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前 言 (第 4 版 ) 


编写 目的 与 背景 
众所周知 ， 当 前 社会 需求 和 高 校 课程 设置 严重 脱节 ， 一 方面 企业 找 不 到 可 迅速 上 手 的 人 才 ， 另 一 


方面 大 学 生 就 业 难 。 如 果 有 一 些 面向 工作 应 用 的 案例 参考 书 ， 让 大 学 生得 以 参考 ， 并 能 亲手 去 做 ， 势 
必 能 缓解 这 种 矛盾 。 本 书 就 是 这 样 一 本 书 : 项 目 开发 案例 型 的 、 面 向 工作 应 用 的 软件 开发 类 图 书 。 编 


最 后 ， 丛 书 第 1 版 于 2008 年 6 月 出 版 ， 并 于 2011 年 和 2013 年 进行 了 两 次 改版 升级 ， 因 为 编写 细 


有 展 ， 易 学 实用 ， 配 备 全 程 视频 讲解 等 特点 ， 备 受 读者 瞩目 ， 从 书 累计 销售 20 多 万 册 ， 成 为 近年 来 最 受 
欢迎 的 软件 开发 项 目 案例 类 从 书 之 一 。 


转眼 5 年 已 过 ， 我 们 根据 读者 朋友 的 反馈 ， 对 丛书 内 容 进 行 了 优化 和 升级 ， 进 一 步 修正 之 前 版 本 
中 的 朴 漏 之 处 ， 并 增加 了 大 量 的 辅助 学 习 资源 ， 相 信 这 套 书 一 定 能 带 给 您 惊喜 ! 


本 书 特点 


色 微 视频 讲解 


对 于 初学 者 来 说 ， 视 频 讲解 是 最 好 的 导师 ， 它 能 够 引导 初学 者 快速 入 门 ， 使 初学 者 感受 到 编程 的 
快乐 和 成 就 感 ， 增 强 进一步 学 习 的 信心 。 鉴 于 此 ， 本 书 为 大 部 分 章节 都 配备 了 视频 讲解 ， 使 用 手机 扫 
描 正文 小 节 标 题 一 侧 的 二 维 码 ， 即 可 在 线 学 习 项 目 制作 的 全 过 程 。 同 时 ， 本 书 提供 了 程序 配置 使 用 说 
明 的 讲解 视频 ， 扫 描 二 维 码 即 可 进行 学 习 。 


颁 典型 案例 


本 书 案例 均 从 实际 应 用 角度 出 发 ， 应 用 了 当前 流行 的 技术 ， 涉 及 的 知识 广泛 ， 读 者 可 以 从 每 个 案 
例 中 积累 丰富 的 实战 经 验 。 


记 代码 注释 
为 了 便于 读者 阅读 程序 代码 ， 书 中 的 代码 均 提 供 了 详细 的 注释 ， 并 且 整 齐 地 纵向 排列 ， 可 使 读者 
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快速 领略 作者 意图 。 
[9 代码 贴 十 


案例 类 书籍 通常 会 包含 大 量 的 程序 代码 ， 宛 长 的 代码 往往 令 初 学 者 望 而 生 旦 。 为 了 方便 读者 阅读 
和 理解 代码 ， 本 书 避 免 了 连续 大 篇 幅 的 代码 ， 将 其 分 割 为 多 个 部 分 ， 并 对 重要 的 变量 、 方 法 和 知识 点 
设计 了 独 具 特 色 的 代码 贴 士 。 


总 知识 扩展 


为 了 增加 读者 的 编程 经 验 和 技巧 ， 书 中 每 个 案例 都 标记 有 “注意 ”“ 技 巧 ”等 提示 信息 ， 并 且 在 
每 章 中 都 提供 有 一 项 专题 技术 。 


本 书 约定 


由 于 篇 幅 有 限 ， 本 书 每 章 并 不 能 逐一 介绍 案例 中 的 各 模块 。 作 者 选择 了 基础 和 典型 的 模块 进行 介 
绍 ， 对 于 功能 重复 的 模块 ， 由 于 技术 、 设 计 思 路 和 实现 过 程 基本 相同 ， 因 此 没有 在 书 中 体现 。 读 者 在 
学 习 过 程 中 若 有 相关 疑问 ， 请 登录 本 书 官方 网 站 。 本 书 中 涉及 的 功能 模块 在 资源 包 中 都 附带 有 视频 录 
像 ， 方 便 读 者 学 习 。 


适合 读者 


本 书 适合 作为 计算 机 相关 专业 的 大 学 生 、 软 件 开发 相关 求职 者 和 爱好 者 的 毕业 设计 和 项 目 开发 的 
参考 书 。 


本 书 服务 


为 了 给 读者 提供 更 为 方便 快捷 的 服务 ,读者 可 以 登录 本 书 官方 网 站 (www.mingrisoft.com) 或 清华 
大 学 出 版 社 网 站 (www.tup.com.cn), 在 对 应 图 书页 面 下 载 本 书 资源 包 , 也 可 加 入 企业 QQ (4006751066) 
进行 学 习 交流 。 学 习 本 书 时 ， 请 先 扫描 封底 的 二 维 码 ， 即 可 学 习 书 中 的 各 类 资源 。 


本 书 作者 


本 书 由 明日 科技 软件 开发 团队 组 织 编写 ， 主 要 由 张 鲍 、 冯 春 龙 执笔 ， 参 与 本 书 编写 工作 的 还 有 赛 
杜 春 、 王 小 科 、 周 佳 星 、 王 国 辉 、 辛 洪 郁 、 张 宝 华 、 申 小 琦 、 高 春 艳 、 葛 忠 月 、 刘 杰 、 李 磊 、 杨 柳 、 
赵 宁 、 宋 万 勇 、 贾 景 波 、 吕 玉 浴 、 白 宏 健 、 杨 丽 、 隋 妍 妍 、 刘 媛 媛 、 李 春 林 、 李 菁菁 、 何 平 、 张 云 凯 、 
申 野 、 庞 凤 、 胡 冬 、 岳 彩 龙 、 潘 建 狼 、 张 渤 洋 、 梁 英 、 于 水 晶 、 李 雪 、 孙 势 、 卞 防 、 朱 艳 红 、 宋 下 蒙 、 
白浪 松 、 依 莹 莹 、 李 颖 、 王 欢 等 ， 在 此 一 并 感谢 ! 

在 编写 本 书 的 过 程 中 ， 我 们 本 着 科学 、 严 说 的 态度 ， 力 求 精益 求 精 ， 但 错误 、 朴 漏 之 处 在 所 难免 ， 
敬 请 广大 读者 批评 指正 。 

感谢 您 购买 本 书 ， 希 望 本 书 能 成 为 您 的 良师益友 ， 成 为 您 步 入 编程 高 手 之 路 的 踏 脚 石 。 

宝剑 锋 从 磨 研 出 ， 梅 花香 自 苦寒 来 。 祝 读书 快乐 ! 
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52 同城 信息 网 


(Apachet+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


在 全 球 知 识 经 济 和 信息 化 高 速 发 展 的 今天 ， 信 息 化 是 决定 企业 成 
败 的 关键 因素 ， 企 业 需 要 在 网 站 上 发 布 供求 信息 ， 以 促使 企业 在 同 领 
域 中 得 到 突飞猛进 的 发 展 。 

一 个 广泛 的 、 快 速 的 、 自 由 的 信息 交流 平台 ， 为 用 户 带 来 方便 的 
同时 ， 也 会 给 企业 带 来 无 限 商机 。 于 是 ， 以 互联 网 为 基础 的 信息 交流 
平台 ， 即 52 同城 信息 网 出 现 了 。52 同城 信息 网 致力 于 优化 信息 交流 ， 
实现 信息 的 快速 交流 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

使 当前 窗口 承载 框架 页 中 的 超 链接 页 面 

Hi 如 何 自 动 计算 以 系统 日 期 为 基数 的 相对 日 期 

由 do…while 循环 语句 的 应 用 

让 查询 关键 字模 红 技 术 

让 在 Windows 操作 系统 下 搭建 PHP 开发 环境 

”在 Windows 操作 系统 下 创建 数据 库 和 数据 表 

”PowerDesigner 建 模 的 应 用 

Mm 单元 测试 技术 

Wm 框架 技术 在 Web 网 站 中 的 应 用 

MW 如 何 发 布 网 站 

Mm 表单 数据 的 两 种 提交 方式 

MW 应 用 phpMyAdmin 工具 创建 和 删除 数据 库 、 数 据 表 





ok 
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1.1 开发 背景 


回 :人 
视频 讲解 

X XX 信息 科技 有 限 公司 是 一 家 以 整合 渠道 资源 为 主 的 高 科技 公司 。 为 了 扩大 企业 规模 ， 增 强 企 
业 的 竞争 力 , 该 公司 决定 向 多 元 化 发 展 , 借助 mternet 在 国内 的 快速 发 展 , 聚集 部 分 资金 投入 网 站 建设 ， 
为 企业 和 用 户 提供 综合 信息 服务 ， 以 向 企业 提供 有 偿 信息 服务 为 盈利 方式 ， 打 造 一 个 全 新 的 供求 信息 
网 。 例 如 ， 提 供 企业 广告 、 发 布 各 类 免费 供求 信息 、 发 布 企业 付费 信息 等 服务 方式 。 现 需要 委托 其 他 
单位 开发 一 个 综合 信息 网 站 。 











1.2 系统 分 析 


1.2.1 需求 分 析 


对 于 信息 网 站 来 说 ， 用 户 的 访问 量 是 至 关 重 要 的 。 如 果 网 站 的 访问 量 很 低 ， 那 么 就 很 少 有 企业 会 
要 求 为 他 提供 有 偿 服 务 ， 也 就 没有 利润 可 言 了 。 因 此 ， 信 息 网 站 必须 为 用 户 提供 大 量 的 、 免 费 的 、 有 
价值 的 信息 才能 够 吸引 用 户 。 为 此 ， 网 站 不 仅 要 为 企业 提供 各 种 有 偿 服 务 ， 还 需要 额外 为 用 户 提供 大 
量 的 无 偿 服务 。 通 过 与 企业 的 实际 接触 和 沟通 ， 确 定 网 站 应 包括 招聘 信息 、 求 职 信息 、 培 训 信息 、 公 
寅 信息 、 家 教 信息 、 车 辆 信息 、 物 品 求购 、 物 品 出 售 、 求 竞 出 竞 、 寻 求 合作 、 企 业 广 告 等 服务 。 
通过 实际 调查 ， 要 求 供求 信息 网 具有 以 下 功能 : 
界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵 活 ， 树 立 企业 形象 。 
实现 强大 的 供求 信息 查询 ， 支 持 模 糊 查询 。 
用 户 不 需要 注册 ， 便 可 免费 发 布 供求 信息 。 
免费 发 布 的 供求 信息 必须 经 后 台 审核 后 才能 正式 发 布 ， 避 免 不 良 信息 。 
支持 海量 数据 录入 。 
由 于 供求 信息 数据 量 大 ， 后 台 应 该 可 以 随时 清理 数据 。 


1.2.2 可行 性 分 析 


根据 《计算 机 软件 产品 开发 文件 编制 指南 》(GB8567 一 1988) 中 可 行 性 分 析 的 要 求 ， 制 定 可 行 性 
研究 报告 如 下 。 


1. 引言 

(1) 编写 目的 。 

为 了 给 企业 的 决策 层 提供 是 否 进 行 项 目 实施 的 参考 依据 ， 现 以 文件 的 形式 分 析 项 目的 风险 、 项 目 
需要 的 投资 与 效益 。 

(2) 背景 。 

XxX XX 信息 科技 有 限 公 司 是 一 家 以 整合 渠道 资源 为 主 的 高 科技 公司 。 企 业 为 了 不 断 满足 客户 的 需 
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求 ， 为 达到 企业 在 同行 业 领域 中 的 领先 地 位 ， 现 需要 委托 其 他 公司 开发 一 个 综合 信息 网 ， 项 目 名 称 为 
52 同城 信息 网 。 


2. 可 行 性 研究 的 前 提 


(1) 要 求 。 

52 同城 信息 网 要 求 能 够 提供 信息 搜索 、 信 息 定位 描 红 、 发 布 免费 信息 、 发 布 付费 信息 、 发 布 企 业 
广告 和 对 各 类 发 布 的 信息 进行 审核 、 删 除 、 检 索 等 功能 。 

(2) 目标 。 

52 同城 信息 网 的 主要 目标 是 提供 强大 的 搜索 功能 、 准 确 的 信息 描 红 定 位 功能 、 付 费 信息 的 管理 功 
能 、 免 费 信息 的 审核 和 删除 功能 。 

(3) 条 件 、 假 定 和 限制 。 

项 目 需要 在 两 个 月 内 交付 用 户 使 用 。 系 统 分 析 师 需要 3 天 内 到 位 ， 用 户 需要 4 天 时 间 确 认 需 求 分 
析 文 档 。 去 除 员工 两 个 月 的 正常 休息 日 16 天， 那么 程序 开发 人 员 需 要 在 1 个 月 零 几 天 的 时 间 内 进行 系 
统 设 计 、 程 序 编码 、 系 统 测试 、 程 序 调试 和 网 站 部 署 工作 。 

(4) 评价 尺度 。 

根据 用 户 的 要 求 ， 系 统 应 以 搜索 引擎 为 主 ， 对 于 发 布 的 供求 信息 应 能 及 时 准确 地 保存 、 审 核 、 查 
询 、 描 红 定 位 。 由 于 用 户 存 在 多 个 营业 点 ， 系 统 应 具有 局 域 网 操作 的 能 力 ， 在 多 个 营业 点 同时 运行 系 
统 时 ， 系 统 中 各 项 操作 的 延 时 不 能 超过 10 秒 钟 。 此 外 ， 在 系统 出 现 故障 时 ， 应 能 及 时 进行 恢复 。 


3. 投资 及 效益 分 析 


(1) 支出 。 

根据 系统 的 规模 及 两 个 月 的 项 目 开 发 周期 ， 公 司 决定 投入 5 个 人 。 因 此 ， 公 司 将 直接 支付 8 万 元 
的 工资 及 各 种 福利 待遇 。 在 项 目 安装 及 调试 阶段 ， 用 户 培 训 、 员 工 出 差 等 费用 支出 需要 2 万 元 。 在 项 
目 维护 阶段 预计 需要 投入 2 万 元 的 资金 ， 累 计 项 目 投入 需要 12 万 元 资金 。 

(2) 收益 。 

用 户 提供 项 目 资金 30 万 元 。 对 于 项 目 运行 后 进行 的 改动 ,采取 协商 的 原则 根据 改动 规模 额外 提供 
资金 。 因 此 从 投资 与 收益 的 效益 比 上 ， 公 司 可 以 获得 18 万 元 的 利润 。 

项 目 完成 后 ， 将 给 公司 提供 资源 储备 ， 包 括 技术 、 经 验 的 积累 ， 以 后 再 开发 类 似 的 项 目 时 ， 可 以 
极 大 地 缩短 项 目 开发 周期 。 


4. 结论 


根据 上 面 的 分 析 ， 技 术 上 不 会 存在 问题 ， 因 此 项 目 延 期 的 可 能 性 很 小 ;在 效益 上 ， 公 司 投入 5 个 
人 、 两 个 月 的 时 间 获 利 18 万 元 ， 比 较 可 观 ; 在 公司 今后 的 发 展 上 ， 可 以 储备 网 站 开发 的 经 验 和 资源 ， 
因此 认为 该 项 目 可 以 开发 。 


1.2.3 ”编写 项 目 计划 书 


根据 《计算 机 软件 产品 开发 文件 编制 指南 》(GB8567 一 1988) 中 的 项 目 开 发 计划 要 求 ， 结 合 单位 
实际 情况 ， 设 计 项 目 计划 书 如 下 。 
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1. 引言 


(1) 编写 目的 。 

为 了 保证 项 目 开 发 人 员 按 时 保质 地 完成 预订 目标 ， 更 好 地 了 解 项 目 实际 情况 ， 按 照 合 理 的 顺序 开 
展 工作 ， 现 以 书面 的 形式 将 项 目 开发 生命 周期 中 的 项 目 任务 范围 、 项 目 团队 组 织 结构 、 团 队 成 员 的 工 
作 责 任 、 团 队 内 外 沟通 协作 方式 、 开 发 进度 、 检 查 项 目 工作 等 内 容 描述 出 来 ， 作 为 项 目 相 关 人 员 之 间 
的 共识 和 约定 以 及 项 目 生 命 周 期 内 的 所 有 项 目 活动 的 行动 基础 。 

(2) 背景 。 

52 同城 信息 网 是 本 公司 与 X X X 信 息 科技 有 限 公司 签 定 的 待 开 发 项 目 , 网 站 性 质 为 信息 服务 类 型 ， 
可 为 信息 发 布 者 有 偿 或 无 偿 提 供 招 聘 、 求 职 、 培 训 、 求 购 、 公 寓 、 和 车辆、 房屋 和 出 售 等 信息 。 项 目 周 
期 为 两 个 月 ， 项 目 背 景 规划 如 表 1.1 所 示 。 


表 1.1 项 目 背景 规划 
项 目 名 称 签 定 项 目 单位 项 目 承担 部 门 
52 同城 信息 网 开发 部 门 


2. 概述 


(1) 项 目 目标 。 
项 目 目标 应 当 符合 SMART 原则 ， 把 项 目 要 完成 的 工作 用 清晰 的 语言 描述 出 来 。52 同城 信息 网 的 
项 目 目 标 如 下 : 
52 同城 信息 网 主要 用 来 为 用 户 提供 信息 服务 ， 对 于 生活 和 工作 中 的 各 类 信息 都 应 尽 可 能 地 全 部 包 
括 在 内 ， 例 如 公寓 、 求 职 、 招 聘 、 培 训 、 招 商 、 房 屋 、 车 辆 、 出 售 、 求 购 等 信息 。 项 目 发 布 后 ， 要 实 
现 能 够 为 用 户 生 活 和 工作 带 来 极 大 的 方便 并 提高 企业 知名 度 、 为 企业 产品 宣传 节约 大 量 成 本 的 目标 。 
整个 项 目 需要 在 两 个 月 的 期 限 结束 后 ， 交 给 客户 进行 验收 。 
(2) 产品 目标 与 范围 。 
一 方面 ，52 同城 信息 网 能 够 为 企业 节省 大 量 人 力 资源 ， 企 业 不 再 需要 大 量 的 业务 人 员 去 跑 市 场 ， 
间接 为 企业 节约 了 成 本 。 另 一 方面 ，52 同城 信息 网 能 够 收集 海量 供求 信息 ,将 会 有 大 量 用 户 访问 网 站 ， 
有 助 于 提高 企业 知名 度 。 
(3) 应 交付 成 果 。 
项 目 开 发 完成 后 ， 交 付 的 内 容 如 下 : 
回 ”以 资源 包 的 形式 交付 52 同城 信息 网 的 源 程序 、 网 站 数据 库 文件 、 系 统 使 用 说 明 书 。 
回 客户 方 应 用 自己 的 服务 器 ， 因 此 需要 乙方 架设 Apache 服务 器 、 安 装 PHP 开发 环境 、 协 助 甲 
方 购买 域名 ， 将 开发 的 52 同城 信息 网 发 布 到 互联 网 上 运行 。 

回 ”网 站 发 布 到 互联 网 上 以 后 ， 进 行 后 期 的 6 个 月 无 偿 维护 与 服务 ， 超 过 6 个 月 后 进行 网 站 有 偿 
维护 与 服务 。 

(4) 项 目 验收 方式 与 依据 。 

项 目 验收 分 为 内 部 验收 和 外 部 验收 两 种 方式 。 在 项 目 开发 完成 后 ， 首 先进 行内 部 验收 ， 由 系统 测 
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试 员 根 据 用 户 需 求 和 项 目 目标 进行 
验收 。 项 目 在 通过 内 部 验收 后 交 给 
用 户 进行 验收 ， 验 收 的 主要 依据 为 
需求 规格 说 明 书 。 

3. 项 目 团 队 组 织 


(1) 组 织 结构 。 

为 了 完成 52 同城 信息 网 的 项 目 
开发 ， 公 司 组 建 了 一 个 临时 的 项 目 
团队 ， 由 项 目 经 理 、 系 统 分 析 师 、 


项 目 经 理 


PHP 开发 工程 师 、 网 页 设计 师 和 系 系统 分 析 师 PHE 开 发 工程 师 网 页 设计 师 系统 测试 员 
统 测试 员 构成 ， 如 图 1.1 所 示 。 图 1.1 项 目 团队 组 织 结构 图 
(2) 人 员 分 工 。 
为 了 明确 项 目 团队 中 每 个 人 的 任务 分 工 ， 现 制定 人 员 分 工 表 ， 如 表 1.2 所 示 。 
表 1.2 人 员 分 工 表 
姓 名 工作 描述 
负责 项 目的 审批 、 决 策 的 实施 、 项 目的 前 期 分 
张 达 明 | MBA 项 目 开发 部 “| 项 目 经 理 析 、 策 划 、 项 目 开发 进度 的 跟踪 、 项 目 质量 的 


检查 

周 兴 伟 负责 系统 功能 分 析 、 系 统 框架 设计 

分 紫 下 负责 软件 前 后 台 设计 与 编码 

王 惠子 负责 网 页 风格 的 确定 、 网 页 图 片 的 设计 
舒心 恰 对 软件 进行 测试 、 编 写 软件 测试 文档 





1.3 系统 设计 


1.3.1 系统 目标 


根据 需求 分 析 的 描述 以 及 与 用 户 的 沟通 ， 现 制定 网 站 实现 如 下 目标 : 

系统 采用 人 机 对 话 方式 ， 界 面 简洁 、 框 架 清晰 、 美 观 大 方 。 

灵活 快速 地 填写 供求 信息 ， 使 信息 传递 更 快捷 。 

信息 查询 灵活 、 方 便 ， 数 据 存储 安全 可 靠 。 

实现 强大 的 后 台 审核 功能 。 

实现 强大 的 搜索 引擎 ， 支 持 模糊 查询 、 关 键 字 描 红 功 能 等 。 

对 用 户 输入 的 数据 ， 系 统 进行 严格 的 数据 检验 ， 尽 可 能 排除 人 为 的 错误 。 
网 站 最 大 限度 地 实现 易 维护 性 和 易 操 作 性 。 


办 办 办 办 办 多 
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回 ” 为 充分 展现 网 站 的 交互 性 ， 供 求 信息 网 采用 动态 网 页 技术 实现 用 户 信息 在 线 发 布 。 
回 ”具备 完善 的 后 台 管 理 功能 ， 能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 


1.3.2 ”系统 功能 结构 


52 同城 信息 网 前 台 功能 结构 图 如 图 1.2 所 示 。 


1.2 52 同城 信息 网 前 台 功能 结构 图 
52 同城 信息 网 后 台 功 能 结构 图 如 图 1.3 所 示 。 


图 1.3 52 同城 信息 网 后 台 功 能 结构 图 
1.3.3 ”系统 流程 图 
52 同城 信息 网 的 系统 流程 如 图 1.4 所 示 。 
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前 台 显示 信息 





图 1.4 系统 流程 图 
1.3.4 ”系统 预览 


52 同城 信息 网 由 多 个 程序 页 面 组 成 。 下 面 列 出 几 个 典型 页 面 ， 其 他 页 面 参见 资源 包 中 的 源 程序 。 
前 台 首 页 如 图 1.5 所 示 , 该 页 面 用 于 实现 各 类 信息 的 查询 、 企 业 广 告 信息 显示 、 后 台 登 录入 口 等 功 
能 。 搜 索引 擎 页 面 如 图 1.6 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 快速 检索 、 查 询 关键 字 描 红 等 功能 。 






a 
站 和 区 人 


图 1.5 前 台 首页 (资源 包 \TM\Ol\cityinfo\index.php) ”图 1.6 搜索 引擎 (资源 包 \TM\O01\cityinfo\admin\findinfo.php) 


©® 
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发 布 免费 信息 页 面 如 图 1.7 所 示 , 该 页 面 用 于 实现 发 布 分 类 的 免费 信息 功能 。 付费 信 息 管理 页 面 如 
加 1.8 所 示 ， 该 页 面 用 于 实现 付费 信息 分 类 查看 、 付 费 信息 审核 、 付 费 信息 删除 等 功能 。 


与 2 同 姑 信息 网 LL 一 ~ ze 一 52 同 项 信息 网 所 








图 1.7 发 布 免费 信息 (资源 包 \TM\Ol\cityinfoveleasephp) 图 1.8 付费 信息 管理 (资源 包 \TMWlVcityinfoadminfind fufeiphp) 


免费 信息 管理 页 面 如 图 1.9 所 示 , 该 页 面 用 于 实现 免费 信息 分 类 查看 、 免 费 信息 审核 、 免 费 信息 删 
除 等 功能 。 管 理 员 登 录 页 面 如 图 1.10 所 示 ， 该 页 面 用 于 实现 对 管理 员 登 录 的 用 户 名 和 密码 进行 验证 等 
功能 。 







52 同 城 信息 网 后 台 管 理 系统 3 
MW 52 同 城 信息 网 


和 0 和 :内 不 后 和 网》 后 代理 














A 











-本 宙 全市， AE pr 
们 让 x9j，[ 下] 后 和 | EB 
图 1.9 免费 信息 管理 图 1.10 管理 员 登 录 
(资源 包 \TM\Ol\cityinfo\admin\find_mianfeiphp) (资源 包 \TM\O1\cityinfo\admin\login.php》 

1.3.5 ”开发 环境 

在 开发 52 同城 信息 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 

1. 服务 器 端 

操作 系统 :Windows 7/Linux (推荐 )。 

服务 器 : Apache 2.4.18。 

回 PHP 软件 : PHP 7.0.12。 

回 数据库: MySQL 5.5.47。 

回 ”MySQL 图 形 化 管理 软件 : phpMyAdmin-3.5.8。 
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开发 工具 : PhpStorm 2016.3。 
浏览 器 : Google Chrome。 
分 辩 率 : 最 佳 效果 为 1680x1050 像素 。 


客户 端 


浏览 器 :Google Chrome。 
分 辩 率 : 最 佳 效 果 为 1680x1050 像素 。 


1.3.6 文件 夹 组 织 结构 


在 编写 代码 之 前 , 可 以 把 系统 中 可 能 用 到 的 文 
件 夹 先 创建 出 来 (例如 ,创建 一 个 名 为 admin 的 文 
件 夹 ， 用 于 保存 网 站 的 后 台 文件 )， 这 样 不 但 可 以 
方便 以 后 的 开发 工作 ， 也 可 以 规范 网 站 的 整体 架 
构 。 笔 者 在 开发 52 同城 信息 网 时 , 设计 了 如 图 1.11 二 中 才 |- 
所 示 的 文件 夹 组 织 结构 图 。 在 开发 时 ， 只 需要 将 所 
创建 的 文件 保存 在 相应 的 文件 夹 中 即 可 。 


1.4 在 Windows 操作 系统 下 搭建 
PHP 开发 环境 


办 罗 NN 多 凶 提 








网 站 根 目录 

用 于 存储 网 站 后 台 文 件 

用 于 存储 网 站 后 台 使 用 的 图 片 资 源 
用 于 存储 数据 库 连接 文件 

用 于 存储 网 站 使 用 的 CSS 样式 表 文件 
用 于 存储 数据 库 文件 


















视频 潮解” 


PHP 能 否 高 效 、 稳 定 地 运行 依赖 于 服务 器 的 编译 和 执行 ， 本 节 主 要 介绍 如 何在 微软 的 Windows 操 
作 系 统 中 架设 安全 、 可 靠 的 PHP 运行 环境 。 


1.4.1 在 Windows 下 应 用 phpStudy 快速 配置 PHP 开发 环境 


phpStudy 是 PHP 网 页 架 站 工具 组 合 包 ,可 以 将 网 络 上 免费 的 架 站 资源 重新 包装 成 单一 的 安装 程序 。 
它 提 供 了 简易 、 快 速 的 PHP 运行 环境 的 搭建 机 制 ， 读 者 只 需 按照 普通 应 用 软件 的 安装 方式 就 可 以 完成 
Apache+MySQL+PHP+phpMyAdmin 的 安装 与 配置 工作 。 

下 面 以 phpStudy 2016 为 例 来 介绍 phpStudy 的 安装 和 使 用 方法 。 

安装 phpStudy 之 前 应 从 其 官方 网 站 上 下 载 安装 程序 。 下 载 地 址 为 http://www.phpstudy.net/a.php/ 
211.html， 在 下 载 地 址 页 面 找到 如 图 1.12 所 示 的 下 载 链接 ， 单 击 该 链接 即 可 进行 下 载 。 


下 载 地 址 : http:/www.phpstudy.netphpstudy/phpStudy20161103.zip 














图 1.12 phpStudy 下 载 链 接 


在 Windows 操作 系统 下 应 用 phpStudy 快速 配置 PHP 开发 环境 的 操作 步骤 如 下 : 
(1) phpStudy 安装 文件 的 压缩 包 下 载 完成 后 ， 首 先 对 该 压缩 包 进行 解压 缩 ， 然 后 双击 
phpStudy20161103.exe 安装 文件 ， 此 时 弹出 如 图 1.13 所 示 的 对 话 框 。 
© 
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(2) 在 图 1.13 所 示 的 对 话 框 中 单 击 文件 夹 小 图 标 选 择 解压 路 径 ， 并 单 击 OK 按钮 开始 解压 文件 ， 
解压 过 程 如 图 1.14 所 示 。 解 压 文件 完成 后 会 弹出 防止 重复 初始 化 的 确认 对 话 框 ， 如 图 1.15 所 示 。 单 击 
“是 ”按钮 后 进入 phpStudy 的 启动 界面 ， 启 动 完成 后 的 结果 如 图 1.16 所 示 。 








QD 单 击 此 处 选择 存 依 路 往 |- 
可 rei 





鳃 Fanext wn. 
到 下 








Di:\phpstudy 回 单 击 OK 按钮 
Lo%_ JF ere 
图 1.13 phpStudy 解压 对 话 框 图 1.14 解压 文件 进度 条 图 1.15 防止 重复 初始 化 确认 对 话 框 


在 Apache 服务 和 MySQL 服务 启动 成 功 之 后 ， 即 完成 了 phpStudy 的 安装 操作 。 打 开 浏 览 器 , 在 地 
址 栏 中 输入 “http://localhost/” 或 者 “http://127.0.0.1/” 后 按 Enter 键 ， 如 果 运 行 结果 出 现 如 图 1.17 所 示 
的 页 面 ， 则 说 明 phpStudy 安装 成 功 。 
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图 1.16 phpStudy 启动 界面 图 1.17 phpStudy 安装 成 功 运 行 页 面 
1.4.2 PHP 服务 器 的 启动 与 停止 


PHP 服务 器 主要 包括 Apache 服务 器 和 MySQL 服务 器 。 重 新 启动 计算 机 后 , 在 默认 状态 下 , Apache 
服务 和 MySQL 服务 是 停止 的 ， 下 面 介绍 在 phpStudy 中 启动 与 停止 这 两 种 服务 器 的 方法 。 
1. 同时 启动 和 停止 Apache 服务 器 和 MySQL 服务 器 


单 击 phpStudy 快捷 方式 图 标 打开 phpStudy， 打 开 后 的 界面 如 图 1.18 所 示 。 单 击 图 1.18 中 的 “ 启 
动 ”按钮 即 可 同时 启动 Apache 服务 和 MySQL 服务 ， 启 动 后 的 结果 如 图 1.19 所 示 。 
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「 运行 状态 - phpStudy 启 停 「 运行 状态 ThrStuay 启 售 
Apache: m = Et Apache: @ ea 
wa 下 || 一 | #E| 本 | waa © || Em| 本 | ‘ms 
提示 信息 一 一 一 一 一 | [运行 模式 一 一 切换 版 本 
a 个 系统 阳 , 务 
他 非 服务 模式 
应 用 
MySQL 管 理 器 
总 其 他 选项 菜单 
图 1.18 ”phpStudy 的 打开 界面 图 1.19 启动 服务 


如 果 想 要 停止 Apache 服务 和 MySQL 服务 ， 只 需要 单 击 图 1.19 中 的 “停止 ”按钮 即 可 。 另 外 ， 单 
击 图 1.19 中 的 “重启 ”按钮 还 可 以 重启 这 两 种 服务 。 


2. 单独 启动 和 停止 Apache 服务 器 和 MySQL 服务 器 


以 管理 Apache 服务 器 为 例 ， 单 击 phpStudy 启动 界面 中 的 “其 他 选项 菜单 ”按钮 ， 然 后 依次 选择 
“服务 管理 器 ”/“Apache” 选 项 ， 在 弹出 的 选项 菜单 中 即 可 对 Apache 服务 器 进行 启动 、 停 止 或 重启 


等 操作 ， 如 图 1.20 所 示 。 


O wy Homerase 
于 看 phpirfo 
四 phpMyhdmin 
© phosudy BE ; 
PHP 归 及 设 量 ， 
号 
打开 配 重 文件。 
运行 枯 态 Tipstwiy 启 咎 | 铀 抹 开 文 他 时。 


Apache: @ 丽 -md 会 S 行 
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10:808 
10:51:09 
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图 1.20 单独 启动 和 停止 Apache 服务 器 
按照 同样 的 方法 可 以 对 MySQL 服务 器 进行 启动 、 停 止 或 重启 等 操作 ， 如 图 1.21 所 示 。 


3. 设置 开机 自动 启动 服务 

在 phpStudy 的 启动 界面 ， 只 需 选 中 “系统 服务 ” 单 选 按钮 ， 然 后 单 击 “ 应 用 ”按钮 即 可 实现 开机 
自动 启动 服务 的 功能 ， 如 图 1.22 所 示 。 

4. 切换 PHP 版 本 

在 开发 该 项 目 时 使 用 的 PHP 版 本 是 PHP 7.0.12， 在 phpStudy 的 启动 界面 ， 单 击 “ 切 换 版 本 ” 超 链 
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接 ， 然 后 选择 要 切换 的 PHP 版 本 ， 如 图 1.23 所 示 ， 单 击 该 选项 后 会 自动 重启 PHP 服务 器 ， 即 可 实现 
PHP 版 本 的 切换 ， 切 换 PHP 版 本 后 的 启动 界面 如 图 1.24 所 示 。 
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图 1.21 单独 启动 和 停止 MySQL 服务 器 图 1.22 设置 开机 自动 启动 服务 
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全 其 他 选项 菜单 
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图 1.23 选择 要 切换 的 PHP 版 本 图 1.24 切换 版 本 后 重启 PHP 服务 



































1.5 数据库 设计 


1.5.1 数据库 分 析 
本 系统 是 一 个 中 小 型 的 供求 信息 平台 ， 但 是 由 于 平台 会 涉及 海量 数据 ， 因 此 需要 充分 考虑 到 成 本 


全 
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问题 及 用 于 需求 〈 如 跨 平台 ) 等 问题 。 而 MySQL 是 世界 上 最 为 流行 的 开放 源码 的 数据 库 ， 是 完全 网 
络 化 的 、 跨 平台 的 关系 型 数据 库 系统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数 
据 库 。 


1.5.2 ”数据 库 概念 设计 

根据 前 面 对 系 统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实 体 分 别 为 免费 信息 
实体 、 付 费 信息 实体 、 广 告 信息 实体 和 管理 员 实体 。 下 面 分 别 介绍 这 几 个 实体 的 E-R 图 。 

1 免费 信息 实体 


免费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标 题 、 信 息 内 容 、 联 系 人 、 联 系 电 话 、 审 核 状 态 和 发 布 
时 间 属 性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 审核 ,“1” 表 示 “ 是 ”“0” 表 示 “ 和 否 ”。 免 费 信息 实 
体 的 E-R 图 如 图 1.25 所 示 。 


2. 付费 信息 实体 


付费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 发 布 时 间 、 截 止 
时 间 和 审核 状态 属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 付费 ,，“1” 表 示 “ 是 ”“0” 表 示 “ 否 ”。 
付费 信息 实体 的 E-R 图 如 图 1.26 所 示 。 





1.25 ”免费 信息 实体 E-R 图 1.26 付费 信息 实体 E-R 图 
3. 广告 信息 实体 


广告 信息 实体 包括 编号 、 信 息 标 题 、 信 息 内 容 、 发 布 时 间 和 推荐 状态 属性 。 其 中 推荐 状态 属性 用 
来 标识 信息 是 否 在 前 台 显示 ,“1” 表 示 “ 是 ”“0” 表 示 “ 否 ”。 广告 信息 实体 的 E-R 图 如 图 1.27 所 示 。 


4. 管理 员 实体 
管理 员 实体 包括 编号 、 管 理 员 名 和 加 密 密 码 属性 。 管 理 员 实 体 的 E-R 图 如 图 1.28 所 示 。 





管理 员 
Ca) 


图 1.27 广告 信息 实体 E-R 图 图 1.28 管理 员 实 体 E-R 图 
@ 
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1.5.3 创建 数据 库 及 数据 表 


结合 实际 情况 及 对 用 户 需 求 的 分 析 ， 可 知 52 同城 信息 网 中 应 用 的 db_pursey 数据 库 主要 包含 如 下 
4 个 数据 表 ， 如 表 1.3 所 示 。 


表 1.3 db_pursey 数据 库 中 的 数据 表 








表 整 ” 理 说 明 
tb admin utf8 general ci 管理 员 信息 表 
tb_advertising utf8 general ci 企业 广告 信息 表 





tb info utf8 general ci 免费 供求 信息 表 
tb leaguerinfo 付费 供求 信息 表 
各 数据 表 的 表 结 构 如 表 1.4 一 表 1.7 所 示 。 
1. tb_admin (管理 员 信息 表 ) 
管理 员 信息 表 主 要 用 于 存储 管理 员 的 信息 。 该 数据 表 的 结构 如 表 1.4 所 示 。 
表 1.4 管理 员 信息 表 结 构 


名 字 | 类 型 | 整理 | 空 | 区 认 | 额外 | 说 明 
4 | | | 百 | 无 |AuromCREMENT| 自动 
i 无 








2. tb_advertising〈 企 业 广告 信息 表 ) 
企业 广告 信息 表 主 要 用 于 存储 企业 发 布 的 广告 信息 。 该 数据 表 的 结构 如 表 1.5 所 示 。 


表 1.5 企业 广告 信息 表 结构 
















| int(4) 
title | varchar(100) | utf8 general ci 
| OD | utf8 general ci 








content 


fdate 








3. tb_info〈 免 费 供求 信息 表 ) 
免费 供求 信息 表 主 要 用 于 存储 用 户 免 费 发 布 的 供求 信息 。 该 数据 表 的 结构 如 表 1.6 所 示 。 


@_ 
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表 1.6 免费 供求 信息 表 结构 
















































类 型 整 理 空 | 默 认 额 外 说 阴 
id int(4) 否 无 AUTO INCREMENT 自动 编号 
type varchar(30) utf8 general ci 否 无 信息 类 型 
title varchar(50) utf8 general ci 否 无 信息 标题 
content varchar(500) utf8 general ci 否 无 信息 内 容 
linkman varchar(20) utf8 general ci 否 无 联系 人 
tel varchar(30) utf8 general ci 否 无 联系 电话 
checkstate | int(1) 否 0 审核 状态 
edate datetime 否 无 发 布 时 间 











4. tb_leaguerinfo (付费 供求 信息 表 ) 
付费 供求 信息 表 主 要 用 于 存储 付费 的 供求 信息 。 该 数据 表 的 结构 如 表 1.7 所 示 。 
表 1.7 付费 供求 信息 表 结构 


名 字 | 类 型 | 整理 | 空 | 默 认 | 额 外 | 
id it) 
type Varchar(20) utf8 general ci | 否 | 无 | | 
title varchar(50) utf8 general ci | 天 | 于 [| | 
content varchar(500) utf8 general ci | 否 | 无 | | 
linkman Varchar(20) utf8 general ci | 否 | 无 | | 
tel varchar(30) utf8 general ci | 否 | 无 | | 
sdate [date [| 到 | | 
showday | date [| 到 | | 
checkstate | int(1) ss[o | | 








1.6 单元 测试 





在 现代 软件 开发 过 程 中 ， 测 试 不 再 作为 一 个 独立 的 生命 周期 ， 而 是 成 为 与 编写 代码 同步 进行 的 开发 
活动 。 单 元 测试 能 够 提高 程序 员 对 程序 的 信心 ， 保 证 程序 的 质量 ， 加 快 软件 开发 速度 ， 使 程序 易于 维护 。 


1.6.1 单元 测试 概述 


在 程序 设计 过 程 中 会 有 多 种 测试 ， 单 元 测试 只 是 其 中 的 一 种 。 单 元 测试 并 不 能 保证 程序 是 完美 无 
缺 的 ， 但 是 在 所 有 的 测试 中 ， 单 元 测试 是 第 一 个 环节 ， 也 是 最 重要 的 一 个 环节 。 单 元 测试 是 一 种 由 程 
序 员 自行 测试 的 工作 。 简 单 地 说 ， 单 元 测试 就 是 测试 代码 撰写 者 依据 其 所 设想 的 方式 执行 是 否 产生 了 


_ 国 
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预期 的 结果 。 

单元 测试 不 仅 是 无 错 编码 的 一 种 辅助 手段 ， 还 必须 是 可 重复 的 ， 即 无 论 是 在 软件 修改 或 是 移植 到 
新 的 运行 环境 的 过 程 中 都 可 以 进行 单元 测试 。 

与 单元 测试 有 密切 关联 的 开发 活动 包括 代码 走读 〈Code review )、 静 态 分 析 (Static analysis) 和 动 
态 分 析 (Dynamic analysis)。 代 码 走读 就 是 对 软件 的 源 代 码 进行 宏观 阅读 ， 整 理 开 发 思路 。 静 态 分 析 就 
是 对 软件 的 源 代码 进行 研读 ， 查 找 错误 或 收集 一 些 度量 数据 ， 并 不 需要 对 代码 进行 编译 和 执行 。 动 态 
分 析 就 是 通过 观察 软件 运行 时 的 动作 ， 来 提供 执行 跟踪 、 时 间 分 析 以 及 测试 覆盖 度 方面 的 信息 。 


1.6.2 单元 测试 的 优点 


单元 测试 具有 以 下 优点 : 

回 “” 一 种 验证 行为 。 程 序 中 的 每 一 项 功能 都 是 用 测试 来 验证 它 的 正确 性 ， 为 以 后 的 开发 提供 支援 。 
就 算是 开发 后 期 ， 也 可 以 轻松 地 增加 功能 或 更 改 程序 结构 ， 而 不 用 担心 这 个 过 程 中 会 破坏 重 
要 的 东西 ， 而 且 它 为 代码 的 重 构 提 供 了 保障 。 这 样 ， 我 们 就 可 以 更 自由 地 对 程序 进行 改进 。 

回 ”一 种 设计 行为 。 编 写 单元 测试 将 使 用 户 从 调用 者 观察 、 思 考 。 特 别 是 先 写 测试 test-first)， 
迫使 设计 者 把 程序 设计 成 易于 调用 和 可 测试 的 ， 即 解除 软件 中 的 耦合 。 

加 ”一 种 编写 文档 的 行为 。 单 元 测试 是 一 种 无 价 的 文档 ， 它 是 展示 函数 或 类 如 何 使 用 的 最 佳 文档 。 
这 份 文档 是 可 编译 、 可 运行 的 ， 并 且 它 保持 最 新 ， 永 远 与 代码 同步 。 

回 具有 回归 性 。 自 动 化 的 单元 测试 避免 了 代码 出 现 回 归 ， 编 写 完成 之 后 ， 可 以 随时 随地 地 快速 

运行 测试 。 





1.7 前 台 首页 设计 


1.7.1 前 台 首 页 概述 


网 站 主页 是 关于 网 站 的 建设 及 形象 宣传 ， 它 对 网 站 生存 和 发 展 起 着 非常 重要 的 作用 ， 应 该 是 一 个 

信息 含量 较 高 、 内 容 较 丰 富 的 宣传 平台 。52 同城 信息 网 前 台 首 页 主要 包含 以 下 内 容 : 

回 ”网 站 菜单 导航 〈 包 括 公寓 信息 、 招 聘 信息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信息 、 车 
辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启示 等 )。 

回 ”发 布 免费 的 供求 信息 《包括 公寓 信息 、 招 聘 信息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信 
息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启示 等 )。 

回 “推荐 供求 信息 显示 〈 包 括 公寓 信息 、 招 聘 信息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信息 、 
车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启示 等 )， 其 中 ， 付 费 信息 按时 间 顺 序 
降序 排列 ， 免 费 信息 按时 间 顺 序 分 页 显示 。 

回 ”显示 推荐 的 企业 广告 信息 。 

回 ”供求 信息 快速 检索 ， 支 持 模 糊 查询 和 查询 关键 字 描 红 功 能 。 

回 ”后 台 登 录入 口 ， 为 管理 员 进 入 后 台 提供 一 个 入 口 。 
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下 面 看 一 下 本 案例 中 提供 的 前 台 首 页 ， 该 页 面 在 本 书 资源 包 中 的 路 径 为 \TM\01\cityinfo\index.php， 
如 图 1.29 所 示 。 


局 惟 四 二 人 Ea 
有 局 


1 和 于 从 种 引 高 村 下 请 克 了 公 守 sot7-t2-20 
位 于 市 中 心 ， 4 人 得 天! 内 访 及 人间， 五 人 两 ! 亲 隐 秋 标 赤 博 单 人 诬 ， 吉 向 六 单 。 兰 挫 厅 ， 方便 起 局 < 有线 电 

视 ， 新 辣 干 崩 ， 家 柜 ， 床 头 配 ， 夺 呈 ! 可 朋 扳 可 月 六 民 责 ! 内 仁 均 大 有 相对 我 定 工作 扒 太 防 + 宝 让 浊 欧 ， 安 夭 ! 专人 管 

得! 号 吉英 户 令 女人 的 怒 节 加 

联系 人 + 完 双 士 。 。 联 忆 也 和 135cssscolsw 


零 层 础 编程 人 员 的 入 门 导师 一 明日 学 院 


内 容 显 示 区 
和 了。 07-i2-a0 oa 人 


到 和 文通 便利 ， 开 失 好 ， 沈 入 住人 员工 作 科 定 素 大 高 ， 欢 迎 有 禄 定 工作 罕 二 争 太 孩 和 住 , 有 不 良 计 好 者 


[3 





Er 士 联系 友 起 ! otat-eeeoeorn 

nin at 201T12-20 04.20.% 

ei tld et Pra 

a 

可 不 人 : 贡生 。 联 和 中 友 : te 
RR 和 1 日 了 汉 于 本 
省 00 0 灌区 也 近 ,2 可 < 道 便 利 ， 宝生 争 ， 玫 古 ， 角 风 好 ， 有 风干 阿 ， 有 电视 ， 扫 来 妆 术 价 人 全 
网 码 :130000 和 二 化 专人 官 理 ， 

联系 人 : 于 网 般 联 筷 所 却 :FEE999 








1.29 ”52 同城 信息 网 首页 
各 区 域 的 介绍 及 所 对 应 的 PHP 文件 如 表 1.8 所 示 。 
表 1.8 页 面 框架 中 各 区 域 介绍 及 对 应 的 PHP 文件 


名 称 对 应 的 PHP 文件 
导航 栏 提供 查看 各 类 信息 的 超 链接 top.php 
信息 检索 区 企业 广告 显示 及 各 类 信息 检索 | leftphp 


内 容 显示 区 根据 用 户 请 求 显示 相应 内 容 根据 请 求 加 载 相应 的 PHP 文件 ， 默 认 加 载 main php 
版 权 区 显示 版 权 信息 bottom php 
1.7.2 前 台 首页 技术 分 析 


在 前 台 首 页 的 内 容 显示 区 中 主要 显示 付费 以 及 免费 的 公寓 信息 ， 在 查询 免费 的 公寓 信息 时 使 用 
了 分 页 技术 ， 在 进行 分 页 查询 时 使 用 了 limit 关键 字 。limit 是 MySQL 中 的 一 个 特殊 关键 字 。limit 子 
句 可 以 对 查询 结果 的 记录 条 数 进行 限定 ， 控 制 它 输出 的 行 数 。 在 前 台 首 页 中 ， 实 现 分 页 查询 的 关键 


© 
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代码 如 下 : 
<?php 
// 查 询 免费 的 公寓 信息 
$sql=mysqli_query($conn,"select count(*) as total from tb_info where type=' 公 寅 信息 ' and checkstate=1"); 
$info=mysqli_fetch_array($sql); // 将 查询 结果 集 返 回 到 数组 
S$total=$info['total"]; // 获 取 查 询 记录 总 数 
$pagesize=4; /每 页 显示 记录 数 
if ($total<=$pagesizeX{ 
S$pagecount=1; /定义 总 页 数 
》 
if(($total% $pagesize)!=0X{ 
S$pagecount=intval($total/$pagesize)+1; /计算 总 页 数 
Jelse{ 
$pagecount=$total/$pagesize; /计算 总 页 数 
} 
if(!isset($_GET['page'])\{ 
$page=1; /定义 当前 页 
jelsef 
$page=intval($_GET[page]); 1/ 获取 当前 页 
} 
// 查 询 当前 页 中 免费 的 公寓 信息 


S$gsql=mysqli_query($conn,"select * from tb_info where type=' 公 寓 信 息 ' and checkstate=1 order by edate desc 
limit ".($page-1)*$pagesize.",$pagesize"); 
$ginfo=mysqli_fetch_array($gsql); /将 查询 结果 集 返 回 到 数组 
if($ginfo)f 
dof // 循 环 输出 免费 的 公寓 信息 
?> 
/省 略 供求 信息 标题 的 HTML 代码 部 分 
<?php 
}while($ginfo=mysqli_fetch_array($gsq)l)); 
?> 








1.7.3 前 台 首 页 的 实现 过 程 


本 系统 中 所 有 的 前 台 页 面 都 采用 了 二 分 栏 结构 , 分 为 导航 栏 、 信 息 检索 区 、 内 容 显 示 区 和 版 权 区 4 
个 区 域 。 为 了 方便 网 站 的 日 后 维护 ， 将 这 4 个 区 域 形成 单独 的 PHP 文件 ， 然 后 应 用 include 语句 将 这 4 
个 文件 包含 进来 。 前 台 首 页 文件 的 代码 如 下 : 


倒 程 01 代码 位 置 ， 资源 包 \TM\01\cityinfo\index.php 
<table width="920" border="0" align="center" cellpadding="0" cellspacing="0" bgcolor="#FFFFFF"> 
<tr valign="top"> 
<td colspan="2"><?php include("top.php");?></td> <!-- 包 含 导航 文件 --> 
</tr> 
<tr> 


<!-- 包 含 信息 检索 文件 --> 
外 
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<td width="217" valign="top"><?php include("left.php");?></td> 
<!-- 包 含 内 容 显示 文件 --> 
<td width="586" valign="top" bgcolor="#FEFEF6"><?php include("main.php");?></td> 
</tr> 
<tr> 
<td colspan=' 
</tr> 
</table> 


其 中 ， 导 航 文件 top.php 页 中 应 用 了 ul 列表 来 创建 文件 的 超 链接 ， 代 码 如 下 : 
倒 程 02 代码 位 置 ， 资源 包 \TM\01\cityinfo\top.php 


<div class="top"> 
<div> 
<a href="release.php">[ 发 布 信息 ]</a> 
<a href="admin/login.php">[ 进 入 后 台 ]</a> 
</div> 
<div> 
<Ul> 
< 二 一 一 一 一 一 一 一 一 一 加 入 收藏 一 -一 一 一 一 一 一 一 
<li><a href="javascript:try{window.external.AddFavorite('http://localhost/cityinfo','52 同城 信息 网 )}catch(e) 
{alert(' 请 使 用 Ctrl + D 收藏 本 站 ");*> 加 入 收藏 </a></li> 
联系 我 们 一 一 一 一 一 一 一 一 一 一 一 一 -> 
<li><a href="mailto:xor@cityinfo*.com"> 联 系 我 们 </a></li> 
</ul> 
</div> 
</div> 
<div> 
<div class="logo"></div> 
<div class="menu"> 
<ul> 
<li><a href="index.php"> 首 &nbsp; 页 </a></li> 
<li><a href="invitejob.php"> 招 聘 信息 </a></li> 
<li><a href="foster.php"> 培 训 信息 </a></li> 
<li><a href="house.php"> 房 屋 信 息 </a></li> 
<li><a href="seekbuy.php"> 求 购 信息 </a></li> 
<li><a href="seekjob.php"> 求 职 信息 </a></li> 
<li><a href="teaching.php"> 家 教 信息 </a></li> 
<li><a href="car.php"> 车 辆 信息 </a></li> 
<li><a href="sale.php"> 出 售 信息 </a></li> 
<li><a href="recruitbusiness.php"> 招 商 引 资 </a></li> 
<li><a href="search.php"> 寻 物 启示 </a></li> 
</ul> 
</div> 
</div> 
<div class="pcard"></div> 





"><?php include("bottom.php");?></td> <!-- 包 含 版 权 信息 文件 --> 


二 
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1.8 免费 供求 信息 发 布 模块 设计 
1.8.1 免费 供求 信息 发 布 模块 概述 


免费 供求 信息 的 发 布 提供 对 象 为 供求 信息 用 户 ， 是 供求 信息 网 站 非常 重要 的 功能 ， 也 是 供求 信息 
网 站 的 核心 功能 。 

免费 供求 信息 发 布 模块 可 以 完成 11 种 不 同类 别 信息 的 发 布 。 用 户 可 以 根据 自身 需要 将 供求 信息 发 
布 到 相应 的 信息 类 别 中 〈 共 包括 11 个 信息 类 别 : 公寓 信息 、 招 聘 信息 、 求 职 信息 、 培 训 信息 、 家 教 信 
息 、 房 屋 信息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启 示 等 类 型 供求 信息 )。 信 息 发 
布 成 功 后 ， 需 要 管理 员 进行 审核 ， 只 有 审核 成 功 的 信息 才能 显示 在 前 台 相 应 的 信息 类 别 网 页 中 。 免 费 
供求 信息 发 布 的 流程 如 图 1.30 所 示 。 















































图 1.30 免费 供求 信息 发 布 流程 图 


1.8.2 ”免费 供求 信息 发 布 模块 技术 分 析 


本 模块 实现 免费 供求 信息 发 布 功能 ， 主 要 应 用 到 如 下 几 个 函数 。 
1，mysqli_connect() 函 数 


打开 一 个 到 MySQL 服务 器 的 连接 。 如 果 成 功 则 返回 一 个 MySQL 连接 标识 ， 失 败 则 返回 false。 
语法 如 下 : 


mysqli mysqli_connect ( [string server [, string username [, string password [, string dbname [, int port [, string 
socket]Il] ) 


@ 
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mysqli_connect0 函 数 的 参数 说 明 如 表 1.9 所 示 。 
表 1.9 mysqli_connect() 函 数 的 参数 说 明 





说 明 
dbname ”| 连接 的 数据 库 名 称 
| MysQL 服务 器 使 用 的 端口 号 
UNIX 域 socket 






参 。 数 
SeIVeT | MySQL 服务 器 地 址 

usemame | 用 户 名 。 默 认 值 是 服务 器 进程 所 有 者 的 用 户 名 
密码 。 默 认 值 是 空 密码 











password 





2. mysqli_select_db() 函 数 

选择 MySQL 数据 库 。 如 果 成 功 返 回 tue， 失 败 返回 false。 语 法 如 下 : 

bool mysqli_select_db(mysqli link, string dbname) 

其 中 ， 参 数 link 为 必 先 参数， 该 参数 为 应 用 mysqli_connect0 函 数 成 功 连接 MySQL 数据 库 服 务 器 
后 返回 的 连接 标识 ， 参数 dbname 为 必 选 参数 ， 该 参数 为 用 户 指定 要 选择 的 数据 库 名 称 。 


下 面 应 用 mysqli_connect0 函 数 连接 MySQL 服务 器 , 然后 应 用 mysqli_select_db0 函 数 连接 MySQL 
数据 库 ， 代 码 如 下 : 





<?php 

$conn = mysqli_connect("localhost", "root", "root"); 1/ 连接 MySQL 服务 器 
$db=mysqli_select_db($conn,"db_pursey") or die (" 数 据 库 连 接 失败 : " .mysqli_error()); // 连 接 数 据 库 db _pursey 
mysqli_query($conn,"set names utf8"); /采用 utf8 编码 方式 

ir 





3. mysqli_query() 函 数 
mysqli_query0 函 数 用 来 执行 SQL 语句 。 语 法 如 下 : 





mixed mysqli_query(mysdqli link, string query [, int resultmode]) 


其 中 ， 参 数 link 为 必 选 参数 ， 该 参数 为 mysqli_connect0 函 数 成 功 连接 MySQL 数据 库 服 务 器 后 所 
返回 的 连接 标识 ; 参数 query 为 必 选 参数 , 该 参数 为 所 要 执行 的 查询 语句 ; 参数 resultmode 为 可 选 参数 ， 
该 参数 取 值 有 MYSQLI_ USE_RESULT 和 MYSQLI STORE_RESULT。 其 中 MYSQLI STORE RESULT 
为 该 函数 的 默认 值 。 如 果 返 回 大 量 数据 可 以 应 用 MYSQLI_USE_RESULT, 但 应 用 该 值 时 ， 以 后 的 查询 
调用 可 能 返回 一 个 commands out of sync 错误 ， 解 决 办 法 是 应 用 mysqli_free_result0 函 数 释放 内 存 。 

如 果 SQL 语句 是 查询 指令 select, 成 功 则 返回 查询 结果 集 , 否则 返回 false; 如 果 SQL 语句 是 insert、 
delete、update 等 操作 指令 ， 成 功 则 返回 tue， 否 则 返回 false。 

4. date() 函 数 


date0 函 数 主 要 用 于 格式 化 一 个 本 地 时 间 / 日 期 。 语 法 如 下 : 
string date(string format, int timestamp) 


该 函数 返回 将 参数 timestamp 按照 指定 格式 格式 化 而 产生 的 字符 串 , 其 中 参数 timestamp 是 可 选 的 ， 


全 
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默认 值 为 tme0， 即 如 果 没 有 给 出 时 间 戳 ， 则 使 用 本 地 当前 时 间 。 
date() 函 数 的 参数 format 的 格式 化 选项 如 表 1.10 所 示 。 


参 


数 


表 1.10 参数 format 的 格式 化 选项 
说 明 





小 写 的 上 午 和 下 午 值 ， 返 回 值 为 am 或 pm 





大 写 的 上 午 和 下 午 值 ， 返 回 值 为 AM 或 PM 





Swatch Intemet 标准 时 ， 返 回 值 为 000 一 999 





月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 为 01 一 31 





星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 为 Mon 一 Sun 





只 | 二 | 局 | 呈 | 双 ||= 


a 


wl" 


月 份 ， 完 整 的 文本 格式 ， 返 回 值 为 January 一 December 

小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 

小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 0 一 23 

小 时 ，12 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 01 一 12 

小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 00 一 23 

有 前 导 零 的 分 钟 数 ， 返 回 值 为 00 一 59 

判断 是 否 为 夏令 时 ， 返 回 值 如 果 是 夏令 时 为 1， 否 则 为 0 

月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 为 1~31 

星期 数 ， 完 整 的 文本 格式 ， 返 回 值 为 Sunday 一 Saturday 
判断 是 否 为 半年 ， 返 回 值 如 果 是 头 年 为 1， 否则 为 0 

数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 为 01 一 12 

3 个 字母 缩写 表示 的 月 份 ， 返 回 值 为 Jan 一 Dec 

数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 

与 格林 威 治 时 间 相差 的 小 时 数 ， 例 如 0200 

RFC 822 格式 的 日 期 ,例如 ，Thu，21 Dec 2000 16: 01: 07 +0200 
秒 数 ， 有 前 导 零 ， 返 回 值 为 00~59 

每 月 天 数 后 面 的 英文 后 级，2 个 字符 ， 例 如 st、nd、rd 或 者 也 。 可 以 和 j 一 起 使 用 
指定 月 份 所 应 有 的 天 数 


























本 机 所 在 的 时 区 





从 UNIX 纪元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 





星期 中 的 第 几 天 ， 数 字 表示 ， 返 回 值 为 0 一 6 





ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 





4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 





Nl <|a|z cs|- 


2 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
年 份 中 的 第 几 天 ， 返 回 值 为 0 一 366 





N 


局 








时 差 偏 移 量 的 秒 数 。UTC 西边 的 时 区 偏 移 量 总 是 负 的 ，UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 返 回 值 为 


-43200 一 43200 
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本。 注意 有 效 的 时 间 惟 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54 到 2038 年 1 月 19 
日 03:14:07 (此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 )。 在 Windows 系统 中 此 范围 限制 为 从 


1970 年 1 月 1 日 到 2038 年 1 月 19 日 。 


例如 ， 应 用 date0 函 数 格式 化 一 个 日 期 ， 并 输出 日 期 的 值 。 代 码 如 下 : 


<?php 
echo date("y/m/d"); 
?> 


结果 为 : 17/12/11。 


1.8.3 ”免费 供求 信息 发 布 模块 的 实现 过 程 


国 。 免费 供求 信息 发 布 使 用 的 数据 表 : tb info 
用 户 通过 单 击 前 台 页 面 导 航 栏 中 的 “我 要 发 布 ” 超 链接 ， 进 入 信息 发 布 页 面 ， 如 图 1.31 所 示 。 程 
序 会 先 验证 用 户 是 否 输入 信息 ， 若 验证 失败 ， 则 返回 信息 发 布 页 面 ， 进 行 相应 提示 ;， 若 验 证 成 功 ， 则 


向 数据 库 中 插入 记录 ， 完 成 发 布 操作 。 
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联系 人 : 了 末 二 
ES 


和 


图 1.31 免费 供求 信息 发 布 网 页 


在 功能 导航 页 top.php 中 添加 “发 布 信息 ” 超 链 接 。 代 码 如 下 : 
倒 程 03 ”代码 位 置 : 资源 包 \TM\01\cityinfo\top.php 





<a href="release.php">[ 发 布 信息 ]</a> 


在 信息 发 布 页 面 选择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 为 了 避免 用 户 添加 空 信息 ， 
在 单 击 “ 发 布 信息 ”按钮 时 ， 应 用 JavaScript 脚本 自 定义 一 个 checkform0 函 数 ， 验 证 提交 的 表单 各 元 


素 是 否 为 空 值 ， 如 果 为 空 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。 代 码 如 下 : 


® 
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倒 程 04 ”代码 位 置 : 资源 包 \TM\01\cityinfo\release_content.php 





<script language="javascript"> 


function checkform(formj{ // 自 定义 一 个 JavaScript 函数 checkform() 
for(i=0;i<form.length:i++}{ /应 用 for 循环 语句 检索 form 表单 元 素 的 值 是 否 为 空 
if(form.elements[i].value=="™")}{ /如 果 form 表单 中 某 个 元 素 值 为 空 
alert(" 请 将 发 布 信息 填写 完整 ! "); // 弹 出 提示 信息 
form.elements[i] .focus(); // 将 焦点 的 值 定位 到 为 空 值 的 表单 元 素 
return false; 1/ 返回 焦点 
} 
} 
</script> 





创建 与 数据 库 db_pursey 的 连接 ， 代 码 如 下 : 


倒 程 05 代码 位 置 ， 资源 包 \TM\01\cityinfo\conn\conn.php 
<?php 
// 连 接 MySQL 服务 器 
@ ”$conn=mysqli_connect("localhost","root","root") or die(" 数 据 库 服 务 器 连接 错误 ".mysqli_error()); 
@ mysqli_select_db($conn,"db_pursey") or die(" 数 据 库 访问 错误 ".mysqli_error());// 连 接 MySQL 数据 库 
db_pursey 
© mysqli query($conn,"setnames utf8"); /采用 utf8 编码 方式 
?> 








< 代码 贴 十 
@ mysqli connectO 函 数 : 连接 MySQL 服务器， 详细 讲解 参见 1.8.2 节 。 
@ mysqli_select db0 函 数 : 连接 MySQL 数据 库 文 件 ， 详 细 讲 解 参见 1.8.2 节 。 
@ mysqli queryO 函 数 : 用 来 向 数据 库 服务 器 发 送 编码 方式 ， 详 细 讲 解 参见 1.8.2 节 。 


提交 表单 信息 到 数据 处 理 页 ， 应 用 insert…into 语句 向 免费 供求 信息 表 中 添加 供求 信息 。 如 果 信息 
添加 成 功 ， 则 弹出 成 功 的 提示 信息 ; 否则 弹出 失败 的 提示 信息 。 代 码 如 下 : 


倒 程 06 ”代码 位 置 ， 资源 包 \TM\01\cityinfo\release_ok.php 


<?php 

header ("Content-type: text/html; charset=utf-8"); // 设 置 文件 编码 格式 
include("conn/conn.php"); /J 连接 数据 库 文件 
date_default_timezone_set("PRC"); // 设 置 时 区 
$type=$_POST[type]; // 获 取信 息 类 型 
Stitle=$_POSTT'title']; 1/ 获 取信 息 标 题 
$content=$_POST[content]; /获取 信息 内 容 
$linkman=$_POST[Ilinkman1]; /获取 联系 人 
S$tel=$_POST['tel]; // 获 取 联 系 电话 


@ $edate=date("Y-m-d Hi:i:s"); 1/ 获 取 发 布 时 间 
© $sq=mysqli_query($conn,"insert into tb_info(type,title,content,linkman,tel,checkstate,edate) 
values('$type','$title','$content','$linkman','$tel',0,'$edate')"); // 将 免费 的 供求 信息 添加 到 数据 表 中 





if($sqlX{ // 如 果 添 加 操作 成 功 ， 则 弹出 提示 信息 
echo "<script>alert( 恭喜 您 ， 信 息 发 布 成 功 ! ');window.location.href='release.php';</script>"; 
jelsef // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 


© 
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echo "<script>alert(' 对 不 起 ， 信 息 发 布 失败 ! ');history.back();</script>"; 
} 


?> 





< 代码 贴 填 
@ date0 函 数 : 格式 化 一 个 本 地 时 间 / 日 期 ， 详 细 讲解 参见 1.8.2 节 。 
@ insert into: 向 指定 的 数据 表 中 添加 数据 信息 。 


1.9 信息 检索 模块 设计 





和 


1.9.1 信息 检索 模块 概述 


信息 检索 是 对 已 存在 于 数据 库 中 的 数据 按 条 件 进行 筛选 浏览 ， 是 查看 历史 信息 和 确认 数据 操作 最 
为 快速 、 有 效 的 办 法 。 信 息 检 索 模 块 主要 通过 选择 信息 类 型 和 输入 查询 关键 字模 糊 查询 供求 信息 资源 ， 
并 输出 查询 结果 。 考 虑 到 供求 信息 的 信息 量 较 大 ， 因 此 本 模块 对 与 查询 关键 字 相 匹配 的 查询 结果 进行 
描 红 ， 从 而 方便 用 户 的 浏览 。 信 息 检 索 模块 的 示意 图 如 图 1.32 所 示 。 





设置 查询 条 件 











图 1.32 信息 检索 模块 的 示意 图 


1.9.2 信息 检索 模块 技术 分 析 


在 对 数据 进行 查询 后 ， 最 终 需 要 将 查询 结果 显示 在 页 面 中 反馈 给 浏览 者 。 在 PHP 中 ， 查 询 结 果 的 
显示 方式 有 很 多 种 ， 最 常用 的 就 是 表格 显示 方式 。 因 为 采用 这 种 方式 显示 的 数据 条 理 清晰 、 简 洁 明了 。 


Re 
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在 利用 表格 显示 查询 结果 时 ， 通 常 是 将 查 
询 结果 保存 在 结果 集中 ， 然 后 需要 使 用 do… 
while 循环 将 其 查询 结果 显示 出 来 。 需 要 注意 的 
是 ， 需 要 先 判断 查询 结果 是 否 为 空 ， 只 有 查询 
结果 不 为 空 时 ， 才 可 以 使 用 循环 语句 显示 数据 。 
为 了 使 读者 更 好 地 理解 通过 表格 显示 查询 结 
果 ， 下 面 给 出 其 实现 流程 图 ， 如 图 1.33 所 示 。 

考虑 到 用 户 不 可 能 全 面 了 解数 据 表 中 的 数 
据 信息 ， 例 如 不 能 确定 所 要 查询 信息 的 内 容 、 
查询 的 主题 等 ， 这 时 就 需要 使 用 like 进行 模糊 
查询 。like 关键 字 需 要 使 用 通配符 在 字符 串 内 查 
找 指定 的 模式 ， 所 以 读者 需要 了 解 通配符 及 其 
含义 。 通 配 符 的 含义 如 表 1.11 所 示 。 


表 1.11 like 关键 字 中 的 通配符 及 说 明 
说 了 明 
由 零 个 或 更 多 字符 组 成 的 任意 字符 串 
任意 单个 字符 
用 于 指定 范围 ， 例 如 [A~F] 表 示 A~F 范围 内 的 任何 单个 字符 
表示 指定 范围 之 外 的 ， 例 如 [^A~F] 表 示 A~F 范围 以 外 的 任何 单个 字符 








图 1.33 信息 检索 模块 流程 图 





如 果 想 查询 包含 “女子 公寓 ”的 信息 ， 可 以 使 用 like 运算 符 配 合 通配符 “%” 完 成 。 其 SQL 语句 
如 下 : 





select * from tb_info where content like '% 女 子 公寓 %'; 





如 果 想 查找 信息 类 型 为 “公寓 信息 ”或 者 内 容 为 “女子 公寓 ”的 信息 时 配合 or 运算 符 来 使 用 。 其 
SQL 语句 如 下 : 





select * from tb_info ”where type=' 公 寓 信息 ' or content like'% 女 子 公寓 %' 
本 模块 实现 付费 信息 与 查询 关键 字 相 匹配 的 信息 的 SQL 语句 如 下 : 


$type=$_POSTTtype']; // 获 取信 息 类 型 
$content=$_POST[content]; // 获 取 查 询 关 键 字 
$sql1=mysqli_query($conn,"select * from tb_leaguerinfo where checkstate=1 and type='$type' and content 
like'%$content%' or title like'%S$content%' or linkman like'%$content%' or tel like'%$content%"); 
$info1=mysqli_fetch_array($sql1); /检索 付费 的 信息 





本 模块 实现 免费 信息 与 查询 关键 字 相 匹配 的 信息 的 SQL 语句 如 下 : 
$sql=mysqli_query($conn,"select * from tb_info where checkstate=1 and type='$type' and content like'%$content%0' 


or title like'% $content%’ or linkman like'%$content%' or tel like’%$content%"); 
$info=mysqli_fetch_array($sql); /检索 免费 的 信息 


局 
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《入 注 意 当 满足 数据 表 中 多 个 字段 中 的 任 一 字段 时 ， 可 以 使 用 or 运算 符 将 多 个 条 件 连 接 。 


另外 ， 由 于 搜索 的 内 容 中 文字 比较 多 ， 为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 信息 ， 所 以 在 搜索 
引擎 中 加 入 了 描 红 功能 。 描 红 功 能 主要 用 str_ireplace0 函 数 实现 , 该 函数 的 具体 讲解 读者 可 参见 本 章 的 
1.14.1 节 。 


1.9.3 “信息 检索 模块 的 实现 过 程 


国信 息 检索 模块 使 用 的 数据 表 : tb info、tb_leaguerinfo 
在 开发 信息 检索 模块 时 ， 由 于 该 网 站 。 La 
含有 大 量 的 数据 信息 ， 为 了 方便 用 户 浏览 
网 站 信息 ， 需 要 添加 复合 条 件 查询 实现 搜 
索 功 能 。 在 信息 检索 区 的 “关键 字 ” 文 本 
框 中 输入 欲 查询 的 关键 字 ， 在 “条 件 ” 下 
拉 列 表 框 中 选择 要 搜索 的 信息 类 型 ， 然 后 
单 击 “ 开 始 搜索 ”按钮 ， 对 指定 条 件 的 记 
录 进 行 检索 并 输出 结果 集 到 浏览 器 ， 同 时 Ce 
为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 。 < TORT SA Has NI rasTT9ssehin FF 
信息 ， 本 模块 对 查询 关键 字 进行 描 红 。 运 a I 
行 结果 如 图 1.34 所 示 。 
信息 检索 页 面 中 所 涉及 的 重要 表单 图 1.34 信息 检索 页 面 的 运行 结果 
元 素 如 表 1.12 所 示 。 





开启 信息 供与 求 新 时 代 ! 


* ee 寓 4 


= 和 
3 和， 拓 可 和 蛋 人间 骨 入 信 






















和 





表 1.12 信息 检索 页 面 所 涉及 的 重要 表单 元 素 


<select name="type"> 

<option value=" 招 聘 信息 ">- 招 聘 信息 -</option> 

<option value=" 求 职 信息 " selected>- 求 职 信息 -</option> 
</select> 


src="Images/btn1.gif onClick="retum chkinput(form)" “开始 搜索 ”按钮 


信息 类 型 








应 用 JavaScript 脚本 自 定义 一 个 chkinput0 函 数 ， 实 现 对 表单 提交 的 信息 进行 验证 。 代 码 如 下 : 


倒 程 07 代码 位 置 : 资源 包 \TM\01\cityinfo\left.php 
<script language="javascript"> 
function chkinput(formX{ // 自 定义 一 个 chkinput() 函 数 
if(form.content.value=="™"){ // 判 断 如 果 查 询 关 键 字 文本 框 等 于 空 


DO 
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alert(" 请 输入 查询 关键 字 ""); // 则 弹出 提示 信息 
form.content.select(); /重新 定位 焦点 
return false; // 返 回 表单 元 素 
} 
} 
</script> 





将 表单 信息 提交 到 数据 处 理 页 ， 连 接 数据 库 文件 ， 接 收 表单 信息 ， 然 后 用 mysqli_query0 函 数 向 服 
务 器 发 送 SQL 语句 ， 检 索 与 查询 关键 字 相 匹配 的 信息 资源 。 代 码 如 下 : 


倒 程 08 ”代码 位 置 ; 资源 包 \TM\01\cityinfo\findinfo.php 


<?php 

include("conn/conn.php"); /| 连接 数据 库 文件 
$type=$_POST[type']; // 获 取信 息 类 型 
$content=$_POST[content]; // 获 取 查 询 关 键 字 


$sql1l=mysqli_query($conn,"select * from tb_leaguerinfo where checkstate=1 and type='$type' and content 
like'%$content%' or title like'%$content%' or linkman like'%$content%' or tel like'%$content%"); 
$info1=mysqli_fetch_array($sql1); /检索 付费 的 供求 信息 
$sql=mysqli_query($conn,"select * from tb _info where checkstate=1 and type='$type' and content 
like'%$content%' or title like'%$content%' or linkman like'%$content%' or tel like'%$content%"); 
$info=mysqli_fetch_array($sql); /检索 免费 的 供求 信息 


?> 





人 
ME 培 明 信息 检索 需要 从 免费 供求 信息 表 tb_info 和 付费 供求 信息 表 tb_leaguerinfo 中 获取 数据 , 因 
此 需要 向 MySQL 服务 器 传递 两 条 SQL 语句 。 


用 do…while 循环 语句 输出 付费 信息 与 查询 关键 字 相 匹配 的 信息 资源 ， 并 用 str_ireplace0 函 数 对 查 
询 关 键 字 实现 描 红 功 能 。 代 码 如 下 : 


倒 程 09 代码 位 置 ， 资源 包 \TM\01\cityinfo\findinfo.php 


<!-- 下 面 输出 的 是 付费 信息 与 查询 关键 字 相 匹配 的 信息 --> 

<?php 

if($info1X{ // 如 果 检 索 到 了 付费 信息 
?> 





<table width="647" height="27" border="0" cellspacing="0" cellpadding="0" bgcolor="#F6F6F6"> 
<tr><td>&nbsp;<span style="color:-#62ab00; font-weight:bolder"> 信 付费 信息 </span></td></tr> 
</table> 
<?php 
dof // 则 用 do…while 循环 语句 输出 付费 信息 
?> 
<table width="647" border="0" cellspacing="0" cellpadding="0" bgcolor="#FAFFF4"> 
<tr> 
<td height="26"> 
<!-- 应 用 str_ireplace() 函 数 对 查询 关键 字 进 行 描 红 --> 
< 上 -对 与 查询 关键 字 所 匹配 的 信息 类 型 进行 描 红 --> 
[<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>", $info1[type]);?>] nbsp; 


@ 
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<!-- 对 与 查询 关键 字 所 匹配 的 信息 标题 进行 描 红 --> 
<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[titie]):?>&nbsp;&nbsp:; 
<!-- 对 与 查询 关键 字 所 匹配 的 发 布 时 间 进行 描 红 --> 
<?php echo str_ireplace($content,"<font color='#FF0000'>".$content."</font>", $info1[edate]);?></td> 
</tr> 
<tr> 
<td height="26">&nbsp;&nbsp; 
<!-- 对 与 查询 关键 字 所 匹配 的 信息 内 容 进 行 描 红 --> 
<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[content]):?></td> 
</tr> 
<tr> 
<td height="26">&nbsp; 联 系 人 : 
<!-- 对 与 查询 关键 字 所 匹配 的 联系 人 进行 描 红 --> 
<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>", $info1llinkman]); ?>&nbsp; 
<!-- 对 与 查询 关键 字 所 匹配 的 联系 电话 进行 描 红 --> 
联系 电话 : <?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[tel]);?> 


= 





<|-- 
</td> 
</ltr> 
</table> 
<?php 
}while(S$info1=mysqli_fetch_array($sql1)); // 循 环 语句 结束 
?> 





5 
入 明 免费 信息 的 输出 方式 与 付费 信息 的 基本 类 似 ， 代 码 部 分 略 ， 详 见 本 书 附 赠 资 源 包 。 


免费 信息 的 输出 方式 与 付费 信息 的 基本 类 似 ， 下 面 给 出 实现 过 程 的 核心 代码 结构 。 
倒 程 10 ”代码 位 置 ， 资源 包 \TM\01\cityinfo\findinfo.php 





< 一 下 面 输出 的 是 免费 信息 与 查询 关键 字 相 匹配 的 信息 -> 
<?php 
if($infoX{ // 如 果 检 索 到 了 免费 信息 

dof // 则 用 do…while 循环 语句 输出 付费 信息 
?> 

// 免 费 信息 的 输出 方式 与 付费 信息 的 类 似 , 代码 略 
<?php 

} while($info=mysqli_fetch_array($sql)); /do…while 循环 语句 结束 

} /所 条 件 语句 结束 

?> 





如 果 在 免费 供求 信息 表 和 付费 供求 信息 表 中 没有 检索 到 与 查询 关键 字 相 匹配 的 数据 ， 则 弹出 提示 
信息 。 代 码 如 下 : 


倒 程 11 代码 位 置 资源 包 \TM\01\cityinfo\findinfo.php 
<table width="647" border="0" cellspacing="0" cellpadding="0"> 
<tr> 


_ 加 ) 
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<td align="center"> 您 检索 的 信息 资源 不 存在 ! </td> 
</tr> 





</table> 





1.10 后 台 首 页 设计 


视频 讲解 





1.10.1 


后 台 首 页 概述 


程序 开发 人 员 在 设计 网 站 后 台 首 页 时 ， 主 要 从 后 台 管理 人 员 对 功能 的 易 操 作 性 、 实 用 性 、 网 站 的 
易 维护 性 考虑 ， 因 此 采用 了 框架 技术 。52 同城 信息 网 后 台 首 页 主要 包含 以 下 内 容 : 


回 


回 
回 
回 


回 


发 布 付费 的 供求 信息 《包括 公寓 信息 、 招 聘 信息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信 
息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启示 等 )， 以 及 付费 信息 的 浏览 、 
审核 及 删除 功能 。 

免费 信息 的 浏览 、 审 核 及 删除 功能 。 

企业 广告 信息 的 发 布 、 浏 览 、 前 台 推荐 显示 、 删 除 功能 。 

“网 站 首页 ” 超 链 接 ， 为 管理 员 进 入 前 台 提供 一 个 入 口 。 

“退出 登录 ” 超 链 接 ， 用 于 注销 当前 用 户 。 


本 案例 中 提供 的 后 台 首 页 如 图 1.35 所 示 。 该 页 面 在 本 书 资源 包 中 的 路 径 为 \TM\01\cityinfo\admin\ 
index.php。 














图 1.35 52 同城 信息 网 后 台 首页 
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1.10.2 后台 首页 技术 分 析 


52 同城 信息 网 后 台 采 用 框架 技术 进行 页 面 布 局 。 所 谓 框架 就 是 网 页 的 各 部 分 为 相互 独立 的 网 页 ， 
又 由 一 个 网 页 将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 者 的 浏览 器 中 ， 重 复出 现 的 内 容 被 
固定 下 来 ， 每 次 浏览 者 发 出 对 页 面 的 请 求 时 ， 只 下 载 
发 生变 化 的 框架 页 面 ， 其 他 子 页 面 保持 不 变 。 SS 

使 用 框架 可 以 将 容器 窗口 划分 为 若干 个 子 窗口 ， 
在 每 个 子 窗口 可 以 分 别 显示 不 同 的 网 页 。 首 先 在 开发 
工具 中 创建 一 个 “ 左 一 中 一 右 ” 的 框架 集 ， 然 后 在 标 矿 
识 @ 中 添加 “上 一 中 一 下 ”的 框架 集 ， 最 后 在 标识 @@ 4 
中 添加 一 个 “ 左 一 右 ” 的 框架 集 ， 从 而 完成 一 个 完整 L 


的 后 台 框架 。 构 建 框架 的 流程 如 图 1.36 所 示 。 
使 用 框架 可 以 非常 方便 地 完成 导航 工作 。 下 面 详 和 | 

















细 介 绍 框架 网 页 的 基本 结构 、 设 置 框 架 集 的 属性 和 设 
置 框架 的 属性 。 


1， 框 架 网 页 的 基本 结构 图 1.36 网 站 后 台 框架 流程 


框架 网 页 通过 一 个 或 多 个 frameset 和 frame 标记 来 定义 。 在 框架 网 页 中 , 将 frameset 标记 置 于 head 
标记 之 后 ， 以 取代 body 标记 的 位 置 ， 当 客户 端 浏 览 器 不 支持 框架 网 页 时 , 还 可 以 使 用 noframes 标记 给 
出 框架 不 能 被 显示 时 的 替换 内 容 。 框 架 网 页 的 基本 结构 如 下 : 


<html> 
<head> 
<title> 基 本 框架 页 </title> 
</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 
<body> 
对 不 起 ! 您 的 浏览 器 不 支持 框架 页 面 的 显示 ! 
</body> 
</noframes> 
</html> 

















2. 设置 框架 集 的 属性 
框架 集 包含 如 何 组 织 各 个 框架 的 信息 ， 可 以 通过 frameset 标记 来 定义 。 框 架 是 按照 行 和 列 来 组 织 
可 以 使 用 frameset 标记 的 下 列 属性 对 框架 的 结构 进行 设置 。 
(1) 左右 分 割 窗口 属性 cols。 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 左右 分 割 窗口 属性 cols 实现 。 其 语法 格 


GD 


屋 
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式 如 下 : 


<frameset cols="Value,value,..."> 
<frame> 
<frame> 

</frameset> 


其 中 ，value 用 于 指定 各 个 框架 的 列 宽 ， 取 值 有 像素 、 百 分 比 〈%) 和 相对 尺寸 (*) 3 种 形式 。 
例如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 列 ， 其 中 第 1 列 占 浏 览 器 窗口 宽度 的 20%， 第 2 列 为 
120 像素 ， 第 3 列 为 浏览 器 窗口 剩余 部 分 的 框架 。 代 码 如 下 : 


<frameset cols="20%,120,” > 
<frame> 
<frame> 

</frameset> 


[ES 如 果 将 cols 属性 设置 为 “*，*，*”， 则 表示 将 窗口 划分 成 3 个 等 宽 的 框架 ; 如 果 将 cols 
属性 设置 为 “*，2*，3*”， 则 表示 左边 的 框架 占 窗口 宽度 的 116， 中 间 的 框架 占 窗口 宽度 的 1/3， 
右边 的 框架 占 窗口 宽度 的 1/2. 


(2) 上 下 分 割 窗 口 属性 rows。 
在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 上 下 分 割 窗口 属性 rows 实现 。 其 语法 格 
式 如 下 : 





<frameset rows="Value,value,..."> 
<frame> 
<frame> 

</frameset> 


其 中 value 用 于 指定 各 个 框架 的 行 高 ， 取 值 有 像素 、 百 分 比 (%) 和 相对 尺寸 (*) 3 种 形式 , 设置 
方法 与 cols 属性 类 似 。 例 如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 行 ， 其 中 的 第 1 行 占 浏览 器 窗口 宽 
度 的 20%， 第 2 行为 120 像素 ， 第 3 行为 浏览 器 窗口 剩余 部 分 的 框架 。 代 码 如 下 : 

<frameset rows="20%,120,*" > 

<frame> 


<frame> 
</frameset> 


(3) 框架 边框 显示 属性 frameborder。 

该 属性 用 于 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框 ， 默 认 值 ) 或 0〈 不 显示 边框 )。 
(4) framespacing。 

该 属性 用 于 指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 如 果 不 设置 该 属性 ， 则 框架 之 间 没有 间隔 。 
(5) 指定 边框 宽度 属性 border。 

该 属性 用 于 指定 边框 的 宽度 ， 只 有 frameborder 属性 为 1 时 有 效 。 


局 
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3. 设置 框架 的 属性 


使 用 <frame> 标 记 可 以 设置 框架 的 属性 ， 包 括 框架 的 名 称 、 框 架 是 否 包 含 滚 动 条 以 及 在 框架 中 显示 
的 网 页 等 。 其 语法 格式 如 下 : 


<frame name=" 框 架 名 称 " src=" 文 件 " frameborder=" 数 值 " scrolling=" 值 " [noresize] > 


name: 指定 框架 的 名 称 。 

src: 指定 在 框架 中 显示 的 网 页 文件 (包括 HTML、ASP 等 网 页 文件 )。 

frameborder: 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 ) 或 0 (不 显示 )。 默 认 值 为 1。 
scrolling: 指定 框架 是 否 包含 滚动 条 。 如 果 将 该 属性 设置 为 yes， 则 框架 包含 滚动 条 ， 若 将 该 
属性 设置 为 no， 则 框架 不 包含 滚动 条 ;如 果 将 该 属性 设置 为 anato， 则 在 需要 时 包含 滚动 条 。 
回 noresize: 可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 。 


1.10.3 后台 首 页 的 实现 过 程 


根据 1.10.1 节 和 1.10.2 节 的 页 面 概述 及 实现 技术 分 析 ， 需 要 分 别 创建 实现 各 区 域 的 PHP 文件 ， 如 
实现 Banner 广告 栏 的 top.php、 功能 导航 栏 的 left.php、 内 容 显示 区 的 main.php 和 页 尾 文件 的 bottom.php 
等 。 实 现 该 系统 后 台 框架 布局 的 完整 代码 如 下 : 


倒 程 12 ”代码 位 置 : 资源 包 \TM\01\cityinfo\admin\index.php 


<frameset rows="*" cols="1*,925,1*" framespacing="0" frameborder="NO" border="0"> 
<frame src="blank.php" name="left" scrolling="NO" noresize> <!-- 设 置 空 框架 页 --> 
<frameset rows="1065,*" cols="*" framespacing="0" frameborder="NO" border="0"> 
<frameset rows="94,590,” cols="*" framespacing="0" frameborder="NO" border="0"> 
<frame src="top.php" name="topFrame" scrolling="NO" noresize> 
<frameset rows="520" cols="236,*" framespacing="0" frameborder="NO" border="0"> 
<frame src="left.php" name="leftFrame" scrolling="NO" noresize> 
<frame src="main.php" name="mainFrame" scrolling="NO" noresize> 
</frameset> 
<frame src="bottom.php" name="bottomFrame" scrolling="NO" noresize> 
</frameset> 
<frame src="blank.php" name="right" scrolling="NO" noresize> 
</frameset> 
<frame src="blank.php"></frameset> <!-- 设 置 空 框架 页 --> 
<noframes><body> 
</body></noframes> 





办 办 多 加 





和 6 注意 在 建设 Web 网 站 时 ， 如 何 让 不 同 分 关 素 的 用 户 才能 看 到 问 页 的 最 住 效果 是 程序 员 在 设计 
之 初 所 要 考虑 的 首要 问题 .为 了 使 屏幕 的 分 辨 率 在 大 于 1024x768 像素 的 设置 时 仍然 处 于 居中 显示 ， 
只 需要 在 设置 框架 布局 时 ， 在 主 框架 两 侧 各 设置 一 个 宽度 相同 的 blank.php 空 页 即 可 。 


四 
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1.11 付费 供求 信息 发 布 模块 设计 


.11.1 “付费 供求 信息 发 布 模块 概述 




















付费 供求 信息 的 发 布 提供 对 象 为 供求 信息 用 户 ， 是 供求 信息 网 
站 非常 重要 的 功能 ， 也 是 供求 信息 网 站 的 盈利 点 。 企 业 或 用 户 可 以 氏 
根据 自身 需要 对 供求 信息 先进 行 付费 ， 付 费 后 由 管理 员 在 后 台 将 供 人 
求 信息 发 布 到 相应 的 信息 类 别 中 ( 共 包 括 11 个 信息 类 别 : 招聘 信息 、 
求职 信息 、 培 训 信息 、 公 寓 信息 、 家 教 信息 、 车 辆 信息 、 物 品 求购 、 用 而 的 信息 
物品 出 售 、 求 竞 出 竞 、 寻 求 合 作 、 企 业 广告 等 类 型 供求 信息 )。 供 求 
信息 成 功 发 布 后 ， 管 理 员 需要 在 后 台 对 发 布 的 供求 信息 进行 审核 ， 
如 果 审 核 通 过 后 ， 则 显示 在 前 台 相 应 的 信息 类 别 网 页 中 。 付 费 供求 a 
信息 发 布 的 流程 如 图 1.37 所 示 。 A 
一 本 ~ 
1.11.2 ”付费 供求 信息 发 布 模块 技术 分 析 Y 
在 前 台 展 示 
付费 供求 信息 与 免费 供求 信息 不 同 的 是 ， 付 费 供求 信息 不 仅 需 生 才 罗 作 和 











要 收取 一 定 的 费用 ， 而 且 还 需要 一 定 的 时 间 限制 ， 例 如 ， 网 站 要 求 。 图 1.37 付费 供求 信息 发 布 流程 图 
一 个 月 〈 按 30 天 计算 ) 每 条 信息 交 10 元 的 信息 费 ， 如 果 用 户 交纳 
20 元 ,那么 信息 显示 的 天 数 就 是 60 天 。 在 前 台 进 行 显示 时 ， 不 需要 管理 员 进行 手动 管理 ， 而 是 通过 程 
序 直接 计算 出 信息 显示 的 截止 时 间 。 
信息 显示 的 截止 时 间 =“ 系 统 当前 日 期 ”+“ 信 息 的 有 效 天 数 〈 与 用 户 交纳 的 信息 费 相 关 )”。 
自动 计算 信息 显示 的 截止 时 间 的 具体 代码 如 下 : 


$days=$_POST['days']; // 通 过 表单 传 值 获取 信息 显示 的 天 数 
$showday=date("Y-m-d",(time()+3600*24*$days)); 1/ 信息 显 示 的 截止 时 间 








/. 
Pn 
天 数 需要 管理 员 手动 添加 。 


1.11.3 ”付费 供求 信息 发 布 模块 的 实现 过 程 


国 付费 供求 信息 发 布 使 用 的 数据 表 : tb leaguerinfo 

用 户 通过 单 击 页 面 导航 区 的 “付费 信息 ” 超 链 接 ， 进 入 付费 信息 发 布 页 面 ， 如 图 1.38 所 示 。 填 写 
真实 有 效 的 付费 信息 ， 单 击 “ 发 布 信息 ”按钮 ， 程 序 会 先 验证 用 户 输入 的 信息 ， 若 验证 失败 ， 则 返回 
信息 发 布 页 面 ， 进 行 相 应 提示 ; 若 验证 成 功 ， 则 向 数据 库 中 插入 记录 ， 完 成 付费 信息 的 发 布 操作 。 


局 
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人 axql。 Fs? [和 
昌 全 业 广 和 时 未 方式 
友 入 HA 

| 











图 1.38 付费 供求 信息 发 布 页 面 运行 结果 
在 左 侧 框架 left.php 页 中 ， 添 加 “付费 信息 ”图 像 域 及 表单 。 代 码 如 下 : 


倒 程 13 ”代码 位 置 ， 资源 包 \TM\01\cityinfo\admin\left.php 


<form name="form1" method="post" action="release_content.php" target="mainFrame"> 

<input name="imageField" type="image" class="input1" src="images/btn_fufei.gif" width="210" height="39" 
border="0"> 
</form> 


单 击 “ 付 费 信息 ”按钮 ， 将 信息 页 release_content.php 中 的 内 容 显示 在 框架 显示 页 mainFrame 中 。 
付费 供求 信息 发 布 页 面 主要 用 于 发 布 付费 的 供求 信息 , 该 页 面 中 所 涉及 的 重要 表单 元 素 如 表 1.13 所 示 。 


表 1.13 付费 供求 信息 页 面 所 涉及 的 重要 表单 元 素 





























名 称 含义 
forml method="post" action="release_ok.php" 表单 
<select name="type"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 -</option> 
“option value=" 求 职 信息 " selected>- 求 职 信息 -</option> 信息 类 型 
<option value=" 寻 人 / 物 启示 ">- 寻 人 / 物 启示 -</option> 
</select> 
flag checkbox class="inputl” Value="1" checked “是 否 付费 ” 复 选 杠 
title text size="50" 信息 标题 
content textarea Cols="55" rows="8" 信息 内 容 
linkman text size="30" 联系 人 
tel text size="30" 联系 电话 
days text 有 效 天 数 
imageField image src="images/fa.jpe" onClick="return checkform(form):" “发 布 信息 ”按钮 
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在 付费 信息 发 布 页 面 选 择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 为 了 避免 用 户 添 加 空 
信息 ， 在 单 击 “ 发 布 信息 ”按钮 时 ， 应 用 JavaScript 脚本 自 定义 一 个 checkform(0) 函 数 ， 验 证 提交 的 表 
单 各 元 素 是 否 为 空 值 ， 如 果 为 空 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。checkform() 
函数 的 代码 部 分 与 例 程 04 相同 ， 这 里 不 再 效 述 。 

提交 表单 信息 到 数据 处 理 页 ， 应 用 insert…into 语句 向 付费 供求 信息 表 中 添加 供求 信息 。 如 果 信息 
添加 成 功 ， 则 弹出 成 功 的 提示 信息 ; 否则 弹出 失败 的 提示 信息 。 代 码 如 下 : 


倒 程 14 代码 位 置 资源 包 \TM\01\cityinfo\admin\release_ok.php 





<?php 

header ("Content-type: text/html; charset=utf-8"); // 设 置 文件 编码 格式 
include("../conn/conn.php"); 1/ 连接 数据 库 文件 
date_default_timezone_set("PRC"); // 设 置 时 区 
$type=$_POST[type'; // 获 取信 息 类 型 
$flag=lisset($_POST[flag])?0:1; // 获 取 付款 状态 
Stitle=$_POSTT'title']; // 获 取信 息 标 题 
$content=$_POST[content]; // 获 取信 息 内 容 
$linkman=$_POST[linkman']; // 获 取 联系 人 
$days=$_POST['days]; // 获 取 发 布 时 间 
Stel=$_POST[tel]; /获取 联系 电话 
$sdate=date("Y-m-d"); // 当 前 系统 时 间 

@ $showday=date("Y-m-d",(time()+3600*24*$days)); // 获 取信 息 的 有 效 时 间 


$sql=mysqli_query($conn,"insert into tb_leaguerinfoltype,title,content,linkman,tel,sdate,showday,checkstate) 
values('$type','$title','$content','$linkman','$tel','$sdate','$showday', $flag)")，// 将 付费 的 供求 信息 添加 到 数据 表 中 


if($sql}{ // 如 果 添 加 操作 成 功 ， 则 弹出 提示 信息 
四 ”echo "<script>alert(' 信 息 发 布 成 功 !'); parent.mainFrame.location.href='release_content.php';</script>"; 
}else{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert(' 信 息 发 布 失败 ! ');history.back();</script>"; 
} 
?> 
Ah 代码 贴 十 


@ date("Ym-d".(time0+3600*24*+$days)): 信息 的 有 效 时 间 = 当 前 期 日 期 + 付费 期 限 。 应 用 time0 函 数 获 取 当 前 日 期 时 
间 锥 ， 付 费 期 限 的 时 间 蕉 等 于 3600 秒 x 24 小 时 x 指定 天 数 ， 并 通过 date0) 函 数 格式 化 为 指定 日 期 格式 。 
@ parentmainFrame location .hrefE-'release_contentphp': 刷新 父 框架 页 release_content.php 中 的 信息 。 


1.12 付费 信息 管理 模块 设计 





1.12.1 付费 信息 管理 模块 概述 
付费 信息 管理 模块 主要 包括 付费 信息 列表 、 付 费 信息 审核 和 付费 信息 删除 等 3 个 功能 , 其 框架 如 图 1.39 


局 
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所 示 。 


























付费 信息 列表 




















付费 信息 删除 















































图 1.39 付费 信息 管理 模块 的 框架 图 
1.12.2 ”付费 信息 管理 模块 技术 分 析 
付费 信息 管理 页 面 在 实现 信息 审核 及 删除 的 功能 时 应 用 到 了 UPDATE 更 新 语句 和 DELETE 删除 语 
句 。 下 面 对 这 两 个 语句 进行 详细 的 讲解。 
1. UPDATE 语句 


UPDATE 语句 用 来 改变 单行 上 的 一 列 或 多 列 的 值 , 或 者 改变 单个 表 中 选 定 的 一 些 行 上 的 多 个 列 值 。 
UPDATE 语句 的 语法 如 下 : 





UPDATE<table_name | view_name> 
SET <column_name>=<expression> 

[...,<last column_name>=<last expression>] 
[WHERE<search_condition>] 


UPDATE 语句 的 参数 说 明 如 表 1.14 所 示 。 


表 1.14 UPDATE 语句 的 参数 说 明 
参数 说 明 
需要 更 新 的 表 的 名 称 。 如 果 该 表 不 在 当前 服务 器 或 数据 库 中 ， 或 不 为 当前 用 户 所 有 ， 这 个 名 称 
可 用 链接 服务 器 、 数 据 库 和 所 有 者 名 称 来 限定 
要 更 新 的 视图 的 名 称 。 通 过 view_name 来 引用 的 视图 必须 是 可 更 新 的 。 用 UPDATE 语句 进行 的 
修改 ， 至 多 只 能 影响 视图 的 FROM 子 句 所 引用 的 基 表 中 的 一 个 
SET 指定 要 更 新 的 列 或 变量 名 称 的 列表 
含有 要 更 改 数据 的 列 的 名 称 。column _name 必须 驻 留 于 UPDATE 子 句 中 所 指定 的 表 或 视图 中 。 
column name 标识 列 不 能 进行 更 新 。 如 果 指定 了 限定 的 列 名 称 ， 限 定 符 必 须 同 UPDATE 子 句 中 的 表 或 视图 的 
名 称 相 匹配 


变量 、 字 面值 、 表 达 式 或 加 上 括号 返回 单个 值 的 subSELECT 语句 。expression 返回 的 值 将 替换 


column name 或 @variable 中 的 现 有 值 
@ 








table_name 





View_name 























expression 
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续 表 
参 。 数 说 明 
指定 条 件 来 限定 所 更 新 的 行 。 根 据 所 使 用 的 WHERE 子 句 的 形式 ， 有 两 种 更 新 形式 : 
WHERE 搜索 更 新 指定 搜索 条 件 来 限定 要 删除 的 行 


定位 更 新 使 用 CURRENT OF 子 句 指定 游标 。 更 新 操作 发 生 在 游标 的 当前 位 置 





al candition | 为 要 更 新 行 指定 需 清 足 的 条 件 。 搜 索 条 件 也 可 以 是 联接 所 基于 的 条 件 。 对 搜索 条 件 中 可 以 包 全 
Searc Leon ton” | 的 亩 词 数量 没有 限制 





人 


下 面 应 用 UPDATE 语句 将 指定 职员 的 工资 进行 调整 ， 例 如 ， 将 “小 璇 ”的 基本 工资 由 2600 元 修 
改 为 3000 元 。 其 SQL 语句 如 下 : 





update tab_laborage set jbgz=3000 where name=" 小 璇 ' 





2. DELETE 语句 


DELETE 语句 实现 删除 数据 记录 。DELETE 语句 的 语法 如 下 : 


DELETE FROM <table_name > 
[WHERE<search-condition>] 








DELETE 语句 的 参数 说 明 如 表 1.15 所 示 。 


表 1.15 DELETE 语句 的 参数 说 明 


参 数 说 了 明 
可 选 的 关键 字 ,可 用 在 DELETE 关键 字 与 目标 table_name、view_name 或 rowset_function_limited 
FROM 
之 间 
table name 





是 要 从 其 中 删除 行 的 表 的 名 称 
<search_condition> | 指定 删除 行 的 限定 条 件 。 对 搜索 条 件 中 可 以 包含 的 谓词 数量 没有 限制 


下 面 应 用 DELETE 语句 删除 “职员 姓名 = 小 璇 ”的 员工 基本 信息 。 其 SQL 语句 如 下 : 
DELETE tab_staffer WHERE ygname=' 小 下 





1.12.3 ”付费 信息 显示 的 实现 过 程 


国 付费 信息 显示 页 面 使 用 的 数据 表 : tb leaguerinfo 

管理 员 在 后 台 功 能 导航 区 的 “付费 信息 显示 方式 ” 栏 中 选择 相应 的 信息 类 别 ， 然 后 按 “ 已 付费 ” 
“未 付费 ”或 “全 部 ”中 的 任意 一 种 状态 对 付费 信息 进行 管理 。 例 如 ， 在 “信息 类 别 ” 下 拉 列 表 框 中 
选择 “公寓 信息 ” 在 “付费 状态 ”选项 组 中 选中 “未 付费 ” 单 选 按钮 ， 单 击 “ 检 索 ” 按 钮 提交 表单 ， 
程序 将 按 指定 条 件 显示 出 符合 条 件 的 所 有 信息 ， 运 行 结果 如 图 1.40 所 示 。 


@ 
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52 同 城 信 筷 网 后 

















图 1.40 付费 信息 显示 页 面 的 运行 结果 


本 系统 提供 了 一 组 单 选 按钮 组 成 的 “付费 状态 ”选项 组 ， 分 为 已 付费 、 未 付费 和 全 部 3 个 选项 。 
选中 “未 付费 ” 单 选 按钮 ， 则 传递 的 值 为 0; 选中 “已 付费 ” 单 选 按钮 ， 则 传递 的 值 为 1; 选中 “全 部 ” 
单 选 按钮 ， 则 传递 的 值 为 ll。 还 提供 了 一 个 下 拉 列 表 框 ， 供 用 户 选择 信息 类 别 。 将 这 些 单 选 按钮 与 下 
拉 列 表 框 都 在 一 个 表单 中 实现 ， 这 样 ， 当 单 击 “检索 ”按钮 提交 表单 后 ， 选 择 的 状态 会 通过 表单 进行 
传递 。 其 表单 代码 如 下 : 


倒 程 15 代码 位 置 ， 资源 包 \TM\01\cityinfo\admin\left.php 
@ <formname="form4" method="post" action="find_fufei.php" target="mainFrame"> 
<tr> 
<td height="65" align="center"> 
四 <fieldset style="height:60;width:190"> 
<legend> 友 付费 状态 </legend> 
<input name="payfor" type="radio" class="input1" value="1"> 已 付费 
<input name="payfor" type="radio" class="input1" value="0" checked> 未 付费 
<input name="payfor" type="radio" class="input1" value="all"> 全 部 
</fieldset> 
</td> 
</tr> 
<tr> 
<td height="34" align="center" background="images/info_d.gif"> 信 息 类 别 : 
<select name="select"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 </option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 </option> 


</select> 
<input type="submit" name="Submit2" value=" 检 索 "> 
</td> 
</tr> 
</form> 
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外 代码 贴 二 

@target= "mainFrame": 指定 链接 的 目标 窗口 ，mainFrame 为 内 容 显示 区 的 框架 名 称 。 另 外 ，target 有 4 个 选项 值 ， 
分 别 介绍 如 下 。 

_blank: 指定 将 链接 的 目标 文件 加 载 到 未 命名 的 新 浏览 器 窗口 中 . 

_parent: 指定 将 链接 的 目标 文件 加 载 到 包含 链接 的 父 框架 页 或 窗口 中 ， 如 果 包 含 链接 的 框 不 是 嵌 套 的 ， 则 链 

接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 。 

_self: 指定 将 链接 的 目标 文件 加 载 到 链接 所 在 的 同一 框架 或 窗口 中 。 

_top: 指定 将 链接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 ， 并 由 此 删除 所 有 框架 。 

@ <fieldset><legend>…</legend></fieldset> 标 签 : 在 字符 集 包 含 的 文本 和 其 他 元 素 外 面 绘制 一 个 方 框 。 该 元 素 是 块 
元 素 ， 必 须 成 对 出 现 。 需 要 注意 的 是 ，fieldset 必须 用 在 form 表单 中 ， 一 个 表单 可 以 有 多 个 <fieldset>…</fieldset>, 每 对 
<fieldset>…</fieldset> 为 一 组 ， 每 组 的 内 容 描 述 使 用 <legend> 设 置 标题 名 称 。 


提交 表单 信息 到 find_fufeiphp 页 ， 程 序 将 按 管 理 员 选择 的 指定 条 件 显示 出 符合 条 件 的 所 有 信息 。 
如 果 管 理 员 选中 “全 部 ” 单 选 按钮 ， 那 么 系统 的 代码 如 下 : 


倒 程 16 代码 位 置 ， 资源 包 \TM\01\cityinfo\admin\find fufeiphp 








<?php 

include("../conn/conn.php"); /连接 数据 库 文件 
$state=isset($_POST[payfor])?$_POST[payfor]:™, // 获 取 管 理 员 选 择 的 付费 状态 
$type=isset($_POST['select])?$_POST[select]:""; // 获 取 管 理 员 选 择 的 信息 类 别 


/如果 管 理 员 选 择 的 付费 状态 为 “全 部 ”， 则 查询 管理 员 指 定 信息 类 别 下 的 所 有 付费 信息 ， 并 按 id 降序 排列 
if($state=="all"){ 
$sql=mysqli_query($conn,"select * from tb_leaguerinfo where type='$type' order by id"); 


else{ 
// 如 果 管 理 员 选 择 的 付费 状态 为 “已 付费 ”或 “未 付费 ”状态 ， 则 按 管理 员 指定 的 条 件 进行 查询 ， 并 按 id 降序 排列 
$sql=mysqli_query($conn,"select* from tb_leaguerinfo where type='$type' and checkstate=$state order by id"); 


} 
$info=mysqli_fetch_array($sql); /执行 SQL 语句 
?> 
人 // 省 略 供求 信息 标题 的 HTML 代码 部 分 
<?php 
if($infoX{ // 如 果 检索 到 了 查询 记录 
dof // 则 应 用 do…while 循环 语句 输出 付费 信息 
if($info['checkstate]==1){ // 如 果 付费 状态 的 值 为 1 
$state1=" 已 付费 "; // 则 将 “已 付费 ” 赋 给 变量 $state1 
}else{ // 如 果 付 费 状 态 的 值 为 0 
$state1=" 未 付费 "; // 则 将 “未 付费 ” 赋 给 变量 $state1 
?> 
寺 一 一 一 检 出 指定 符合 查询 条 件 的 付 正 信息 一 一 一 一 一 一 一 
<tr bgcolor="#FFFFFF"> 


<td align="center"><?php echo S$infor'title];?></td> 

<td align="center">&nbsp;<?php echo S$info['content];?></td> 
<td align="center"><?php echo S$info['linkman'];?></td> 

<td align="center"><?php echo S$info['tel];?></td> 


@ 
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<td align="center"><?php echo S$info[ sdate'];?></td> 

<td align="center"><?php echo $info['showday']:?></td> 

<td align="center" class="style11"><?php echo $state1;?></td> 

<td align="center" bgcolor="#FFFFFF"> 
<a href="statefu_ok.php?id=<?php echo Sinfo[id"];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
审核 </a> 
/<a href="fudel_ok.php?id=<?php echo $info['id];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
删除 </a> 
</td> 
</tr> 
<|-- - -> 
<?php 

}while($info=mysqli_fetch_array($sql)); /ldo…while 循环 语句 结束 

} /所 条 件 语句 结束 





// 如 果 未 检索 到 与 管理 员 指定 条 件 相 匹配 的 记录 ， 则 输出 相关 的 提示 信息 
else 
?> 
<tr align="center" bgcolor="#FFFFFF"> 
<td colspan="8"> 对 不 起 ， 您 检索 的 信息 不 存在 ! </td> 
</tr> 
<?php 


?> 





4 
I 


1.12.4 ”付费 信息 审核 的 实现 过 程 


国 ”付费 信息 审核 使 用 的 数据 表 : tb_leaguerinfo 
经 过 审核 的 信息 说 明 该 信息 为 已 付款 信息 。 如 果 企 业 或 个 人 用 户 已 登录 供求 信息 但 未 直接 付费 ， 


想 要 后 期 付款 ， 那 么 管理 员 进 行 审核 时 ， 信 息 不 能 通过 ， 要 求 必须 进行 付款 ， 经 过 审核 的 信息 才能 在 
前 台 进行 显示 。“ 审 核 ” 超 链接 的 代码 如 下 : 


倒 程 17 代码 位 置 资源 包 \TM\01\cityinfo\admin\find fufeiphp 


<a href="statefu_ok.php?id=<?php echo S$info['id"];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
审核 </a> 


管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 





递 到 数据 处 理 页 statefu_ok.php， 用 UPDATE 语句 将 付费 状态 设置 为 1， 说明 该 信息 已 经 付款 。 数 据 处 


理 页 的 代码 如 下 : 
倒 程 18 ”代码 位 置 ; 资源 包 \TM\01\cityinfo\admin\statefu_okphp 
<?php 
header ("Content-type: text/html; charset=utf-8"); /设置 文件 编码 格式 





@ 
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include("../conn/conn.php"); /| 连接 数据 库 文件 

$id=$_GETrid']: /获取 信息 id 的 值 
$type=$_GET[type']; // 获 取信 息 类 型 
$state=$_GET[state]; // 获 取信 息 付费 状态 

@ $sq=mysqli_query($conn,"update tb_leaguerinfo set checkstate=1 where id=$id"); // 更 新 对 应 付费 信息 的 
状态 为 已 付款 

if($sql)}{ // 如 果 更 新 操作 成 功 


@ ”echo "<script>alert(' 该 信息 已 经 通过 审核 ! 
');window.location.href="find_fufei.php?type=$type&state=$state';</script>"; // 弹 出 操作 成 功 提示 信息 
} 





else{ // 如 果 更 新 操作 失败 
echo "<script>alert(' 该 信息 审核 操作 失败 ! ');history.back();</script>";// 弹 出 操作 失败 信息 
} 
?> 
< 代码 贴 十 


@ update…set: 用 来 修改 指定 表 中 的 数据 。 UPDATE 语句 的 使 用 方法 参见 1.12.2 节 。 

@ type=$type&state=$state: 将 变量 type 与 state 的 值 重新 传 到 find fufeiphp 页 ， 目 的 是 为 了 使 find_fufeiphp 页 中 
的 $type 和 $state 变量 重新 获得 信息 类 型 和 付费 状态 值 (管理 员 选 择 的 检索 条 件 )， 从 而 返回 到 付费 信息 管理 页 ， 并 更 新 
数据 信息 。 如 果 数 据 处 理 页 不 对 这 两 个 变量 进行 传 值 ， 那 么 在 返回 到 付费 信息 管理 页 find fufeiphp 时 将 会 因为 检索 不 
到 变量 的 值 而 出 错 。 


1.12.5 ”付费 信息 删除 的 实现 过 程 


回 付费 信息 删除 使 用 的 数据 表 : tb leaguerinfo 
付费 信息 管理 页 中 “删除 ” 超 链接 的 代码 如 下 : 


倒 程 19 代码 位 置 资源 包 \TM\01\cityinfo\admin\find fufeiphp 
<a href="fudel_ok.php?id=<?php echo S$info['id'];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
删除 </a> 


管理 员 单 击 对 应 主题 信息 后 面 的 “删除 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 
递 到 数据 处 理 页 fudel_ok.php， 用 DELETE 语句 将 id 指定 的 供求 信息 删除 。 数 据 处 理 页 的 代码 如 下 : 


倒 程 20 ”代码 位 置 ， 资源 包 \TM\01\cityinfo\admin\fudel_ok.php 











<?php 

header ("Content-type: text/html; charset=utf-8"); /1 设置 文件 编码 格式 
include("../conn/conn.php"); /连接 数据 库 文件 

$id=$_GET[id]; // 获 取信 息 id 的 值 
$type=$_GET['type']; // 获 取信 息 类 型 
$state=$_GET[state]: // 获 取信 息 付费 状态 
$sql=mysqli_query($conn,"delete from tb_leaguerinfo where id=$id"); /删除 对 应 的 供求 信息 

if($sqlX{ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 


echo "<scripf>alert( 该 信息 已 经 删除 ! ");window.location.href='find_fufei.php?type=$type&state=$state';</script>"; 


多 
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} 
else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert(' 该 信息 删除 操作 失败 ! '");history.back();</script>"; 


} 
?> 


\ 
find fufeiphp， 目 的 是 返回 到 付费 信息 管理 页 ， 查 看 执行 删除 操作 后 的 状态 。 


1.12.6 单元 测试 


在 开发 完 管理 员 模块 后 ， 对 该 模块 进行 单元 测试 。 当 管理 员 对 付费 供求 信息 进行 审核 后 ， 审 核 操 
作成 功 ， 但 却 弹出 如 图 1.41 所 示 的 错误 提示 。 


未 到 站 所 信息 的 错误 提示 










GT 


inea | 
昌 放生 是 素 力 式 
A 

EE 


信息 类 别 为 空 , 说 明 没有 获取 到 值 








1.41 审核 付费 供求 信息 的 错误 提示 


在 图 1.41 中 的 错误 提示 中 可 以 看 出 ， 在 付费 信息 管理 页 的 第 8 行 、 第 9 行 、 第 16 行 和 第 37 行 出 
现 问 题 。 下 面 看 一 下 出 现 问 题 的 这 几 行 代码 : 


$state=$_GET[state]; // 获 取信 息 付费 状态 

$type=$_GET[type']; // 获 取信 息 类 型 

if($state=="all"}{ // 如 果 管理 员 选 择 的 付费 状态 为 “全 部 ”， 则 执行 下 面 的 SQL 语句 
$sql1=mysqli_query($conn,"select count(*) as total from tb_leaguerinfo where type='$type' order by id"); 

Jelse{ /否则 ， 执 行 下 面 的 SQL 语句 


$sql1=mysqli_query($conn,"select count(*) as total from tb_leaguerinfo where type='$type' and checkstate= 
$state order by id"); 
/全 人 和 


S$minfo=mysql_fetch_array($sql1); 


7 人 


Sinfo=mysqli_fetch_array($sql); 


@ 
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从 代码 中 可 以 看 出 ，SQL 语句 的 书写 并 没有 错误 。 根 据 图 1.41 所 示 页 面 的 结果 ， 当 前 信息 类 别 为 
空 ， 则 说 明 管理 员 选 择 的 信息 类 别 没 有 传 过 来 值 。 由 此 可 以 看 出 ， 这 是 由 于 在 执行 审核 后 页 面 重新 刷 
新 了 ， 因 此 检索 不 到 管理 员 选 定 的 查询 条 件 值 。 

“审核 ” 超 链接 的 源 代码 如 下 : 


倒 程 21 ”代码 位 置 ， 资源 包 \TM\01\cityinfo\admin\find fufeiphp 
<a href="statefu_ok.php?id=<?php echo $info['id"];?>"> 审 核 </a> 








审核 处 理 页 的 源 代码 如 下 : 

倒 程 22 代码 位 置 ， 资源 包 \TM\01\cityinfo\admin\statefu_ok.php 

<?php 

$id=$_GET[id]; // 获 取信 息 id 的 值 
S$sql=mysqli_query($conn,"update tb_leaguerinfo set checkstate=1 where id=$id"); /更 新 对 应 付费 信息 的 状态 
为 已 付款 

if($sql){ // 如 果 更 新 操作 成 功 ， 弹 出 提示 信息 


echo "<script>alert(' 该 信息 已 经 通过 审核 ! ');window.location.href='find_fufei.php';</script>"; 
} 


解决 该 问题 的 方法 需要 在 “审核 ” 超 链 接 传 值 时 将 管理 员 选 定 的 “信息 类 型 ” 和 “审核 状态 ”的 
变量 值 一 同 传递 到 数据 处 理 页 ， 当 审核 操作 完成 后 ， 再 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 重 新 
传递 给 付费 信息 管理 页 find_fufei.php 即 可 。 

“审核 ” 超 链接 修改 后 的 代码 〈 加 粗 的 代码 部 分 为 修改 的 代码 部 分 ) 如 下 : 


倒 程 23 代码 位 置 ， 资源 包 \TM\01\cityinfo\admin\find_fufei.php 


<a href="statefu_ok.php?id=<?php echo $inforid];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
审核 </a> 
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| pe 
| 隐 技 瑟 在 全 过 多 个 赤 量 时 ， 交 量 之 间 用 “&” 符 号 分 隔 、 


管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 
递 到 数据 处 理 页 statefu_ok.php。 在 执行 完 更 新 操作 后 ， 需 要 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 
重新 传递 到 付费 信息 管理 页 find_fufeiphp， 加 粗 的 代码 部 分 为 修改 的 代码 部 分 。 


倒 程 24 ”代码 位 置 ， 资源 包 \TM\01\cityinfo\admin\statefu_ok.php 


<?php 

$id=$_GET[id]; // 获 取信 息 id 的 值 
$type=$_GET[type]; // 获 取信 息 类 型 
$state=$_GET['state]; /获取 信息 付费 状态 
$sql=mysqli_query($conn,"update tb_leaguerinfo set checkstate=1 where id=$id"); /更 新 对 应 付费 信息 的 状态 为 已 
付款 

if$sql) // 如 果 更 新 操作 成 功 





局 
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echo "<script>alert(' 该 信息 已 经 通过 审核 ! ');windowlocation.href=find_fufei.php?type=$type&state= 


$state";</script>"; 
} 


// 弹 出 操作 成 功 提示 信息 





1.13 网 站 发 布 


开发 52 同城 信息 网 的 最 终 目 的 是 将 其 发 布 到 Intemet 上 ， 供 用 户 浏览 访问 。 


在 服务 器 上 上 载 网 站 ， 首 先 需 从 服务 商 处 申请 固定 的 人 P 
号 ， 然 后 再 注册 一 个 域名 ， 并 将 域名 指定 到 该 他 地 址 。 在 服 
务 器 上 安装 PHP 的 开发 环境 ， 最 后 上 载 网 站 。 

下 面 以 52 同城 信息 网 为 例 , 来 讲解 在 个 人 服务 器 上 上 载 
网 站 的 过 程 。 

(1) 右 击 “ 网 络 ”， 在 弹出 的 快捷 菜单 中 选择 “属性 ” 
命令 ， 打 开 “ 网 络 和 共享 中 心 ”对 话 框 。 

(2) 单 击 “ 本 地 连接 ”， 在 弹出 的 对 话 框 中 单 击 “属性 ” 
按钮 ， 打 开 “ 本 地 连接 属性 ”对 话 框 ， 如 图 1.42 所 示 。 在 该 
对 话 框 的 “此 连接 使 用 下 列 项 目 ” 列 表 框 中 选中 “Intemet 协 
议 版 本 4〈TCP/IPv4)” 复 选 框 ， 然 后 单 击 “ 属 性 ”按钮 ， 按 
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服务 商 提供 的 指定 他 设置 服务 器 的 人 地 址 ， 如 图 1.43 所 示 。 a 


(3) 安装 PHP 的 开发 环境 ， 参 见 本 章 1.4 节 。 


(4) 将 52 同城 信息 网 上 载 到 服务 器 指定 的 路 径 下 ， 如 图 1.44 所 示 。 
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1.43 “Intemet 协议 版 本 4 (TCPIPv4) 属性 ”对 话 框 图 1.44 上载 $2 同城 信息 网 到 指定 的 路 径 下 


(5) 在 浏览 器 地 址 栏 中 输入 申请 的 域名 ， 即 可 浏览 发 布 的 网 站 。 
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1.14 开发 技巧 与 难点 分 析 


1.14.1 查询 关键 字 描 红 功 能 

















在 52 同城 信息 网 前 台 信 息 检索 过 程 中 体现 了 方便 快捷 的 人 性 化 
原则 ， 为 了 方便 浏览 者 查阅 信息 ， 便 于 查找 与 浏览 者 的 关键 字 相 符合 代 
的 信息 ， 在 搜索 引擎 中 添加 描 红 功能 。 用 户 
查询 关键 字 描 红 是 指 将 查询 关键 字 以 特殊 的 颜色 、 字 号 或 字体 进 
行 标识 ， 这 样 可 以 使 浏览 者 快速 找到 所 需 的 关键 字 ， 方 便 浏 览 者 从 搜 设置 查询 条 件 
索 结果 中 查找 所 需 内 容 。 查 询 关 键 字 描 红 适 用 于 模糊 查询 。 下 面 介绍 ; 
如 何 实 现 查询 关键 字 描 红 。 
本 系统 用 str_ireplace0 函 数 来 蔡 换 查询 关键 字 , 当 显 示 所 查询 的 相 查询 结果 
关 信 息 时 ， 将 输出 的 关键 字 的 字体 替换 为 红色 。 描 红 功 能 的 业务 流程 
如 图 1.45 所 示 。 将 关键 字 描 红 











str_ireplace0 函 数 用 于 将 某 个 指定 的 字符 串 蔡 换 为 另 一 个 指定 的 


字符 串 ， 不 区 分 大 小 写 。 该 函数 的 语法 如 下 人 





mixed str_ireplace(mixed search, mixed replace, mixed subject [, int &count]) 





该 函数 将 所 有 在 参数 subject 中 出 现 的 参数 search 以 参数 replace 取代 ， 参 数 &count 表示 取代 字符 
串 执行 的 次 数 。 
str_ireplace0 函 数 的 参数 说 明 如 表 1.16 所 示 。 


表 1.16 str_ireplace() 函 数 的 参数 说 明 
说 明 
必要 参数 ， 指 定 需要 查找 的 字符 串 
必要 参数 ， 指 定 替换 的 值 
必要 参数 ， 指 定 查找 的 范围 
可 选 参数 ， 获 取 执行 替换 的 数量 


参数 
Search 
Teplace 





Subject 








Count 


Or 
用 str_Ieplace0 函 数 。 

本 系统 应 用 str_ireplace0 函 数 若 换 查询 字符 串 为 红色 的 字符 串 ， 关 键 代码 如 下 ; 

<?php 


include("conn/conn.php"); /1 连接 数据 库 文件 
$content=$_POST[content]; // 获 取 查 询 关 键 字 


局 
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$sql1=mysqliquery($conn,"select * from tb_leaguerinfo where checkstate=1 and type='$type' and content 
like'%$content%' or title like'%$content%' or linkman like'%$content%' or tel like'%$content%"); 
$info1=mysqli_fetch_array($sql1); // 采 用 模糊 信息 资源 查询 
// 下 面 应 用 str_ireplace() 函 数 将 指定 的 查询 关键 字 用 红色 文字 蔡 代 ， 并 输出 蔡 换 后 的 字符 串 

echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[type"]); // 蔡 换 信息 类 型 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[title]); // 蔡 换 信息 标题 为 红色 字体 
echo str_ireplace($content,"<font color='#FF0000'>".$content."</font>",$info1[edate"]);，// 蔡 换 发 布 时 间 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000>"$content "</font>",$info1f content]);// 蔡 换 信 息 内 容 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[linkman1]);// 蔡 换 联系 人 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[te 门 ); // 营 换 联系 电话 为 红色 字体 
?> 





1.14.2 ”表单 数据 的 提交 方式 


获取 表单 元 素 提交 的 值 是 表单 应 用 中 最 基本 的 操作 方法 。 表 单数 据 的 传送 方法 有 get 方法 和 post 
方法 两 种 ,通过 <form> 的 method 属性 来 指定 。 下 面 来 具体 讲解 一 下 这 两 种 方法 在 实际 工作 中 的 应 用 范 
围 和 使 用 技巧 。 


1. 通过 get 方法 提交 数据 
使 用 get 方法 时 ， 表 单数 据 被 当 作 url 的 一 部 分 一 起 传 过 去 。 格 式 如 下 : 


http:Wurl?name1=value1&name2=value2... 


回 url:， 表单 响应 地 址 。 例 如 ，127.0.0.1/index.php。 

加 ”name: 表单 元 素 的 名 称 。 例 如 , <input type="text" name="user">, 这 里 name 的 属性 值 就 是 user。 
通过 name 值 可 以 获取 value 的 属性 值 。 

回 “value: 表单 元 素 的 值 。 例 如 ，<input type="text" name="user" value="mr">， 意 思 是 名 字 叫 user 
的 text 表单 元 素 的 值 为 mr。 











人 
WM 和 表单 元 素 之 间 用 “9” 隔 开 ， 而 多 个 表单 元 素 之 间 用 “&” 隔 开 ， 每 个 表单 元 素 的 格 
式 都 是 “name=value”， 固 定 不 变 。 

PHP 使 用 $_GET 预定 义 变量 自动 保存 通过 get 方法 传 过 来 的 值 ， 使 用 格式 为 : 

$_GET[name] 


这 样 ， 就 可 以 直接 使 用 名 字 为 name 的 表单 元 素 的 值 。 


| 


隔世 有 的 PHP 版 本 中 直接 写 Sname 就 可 以 调用 表单 元 素 的 值 ,这 和 php ini 文 件 的 配置 有 关系 ， 
定位 到 GLOBAL=OMN/OFEF 行 ， 如 果 值 为 ON， 就 可 以 直接 写成 Sname， 反 之 则 不 可 以 。 直 接应 用 表 
单 名 称 十 分 方便 ， 但 也 存在 一 定 的 安全 隐患。 推荐 读者 关闭 GLOBAL 项 。 


使 用 文本 框 传 值 的 程序 中 包含 一 个 文本 框 元 素 。 在 文本 框 中 输入 信息 ， 当 单 击 “提交 ”按钮 时 ， 


_ 回 ) 
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文本 框 内 的 信息 就 会 和 url 一 起 显示 在 地 址 栏 中 。 代 码 如 下 : 
<form name="login" method="get" action="index.php"> 
</form> 
恒生。 坟 计 是 全 用 i 来 进行 信 什 的， -加 入 收藏 交 ”的 功能 就 是 Bct 方法 的 应 用 之 超 要 


将 一 个 带 参 数 的 网 址 加 入 到 收藏 夹 中 ， 只 能 使 用 url, 它 可 以 脱离 表单 的 束缚 . 文字、 图 片 等 都 可 以 
使 用 这 种 方法 来 传 值 。 


2. 通过 post 方法 提交 数据 


get 方法 有 个 最 大 的 缺点 ， 就 是 它 的 信息 是 显示 在 客户 端 浏览 器 上 的 ， 这 使 用 户 的 资料 暴露 无 遗 ， 
而 且 ul 本 身受 长 度 限 制 (1024KB ), 不 能 传输 较 大 的 数据 。 这 时 可 以 选择 post 方法 。 使 用 时 , 将 <form> 
表单 中 的 属性 method 设置 成 post 即 可 。post 方法 不 依赖 url， 所 有 提交 的 信息 在 后 台 传输 ， 不 会 显示 
在 地 址 栏 中 ， 安 全 性 高 ， 而 且 没有 长 度 限制 。 

使 用 PHP 的 $_ POST[mame'] 变 量 可 以 获取 表单 元 素 的 值 ， 格 式 和 $_GET[mame'] 类 似 : 


$_POST[name'] 








例如 ， 使 用 的 post 方 法 返回 文本 框 信息 ， 代 码 如 下 : 





<form name="login" method="post" action="index.php"> 


</form> 





1.15 MySQL 数据 库 技术 专题 


phpMyAdmin 是 众多 MySQL 图 形 化 管理 工具 中 应 用 最 广泛 的 一 种 ， 它 是 基于 PHP 语言 编写 的 ， 
该 工具 是 B/S 结构 、 基 于 Web 跨 平台 的 管理 程序 ， 并 且 支 持 简 体 中 文 ， 可 以 在 官方 网 站 免费 下 载 。 安 
装 后 在 浏览 器 地 址 栏 中 输入 http://127.0.0.1/phpMyAdmin/， 即 可 进入 MySQL 的 管理 界面 。 
phpMyAdmin 为 Web 开发 人 员 提 供 了 类 似 于 Access、SQL Server 的 图 形 化 数据 库 操作 界面 ， 通 过 
该 管理 工具 可 以 进行 绝 大 部 分 的 MySQL 操作， 包括 对 数据 库 及 数据 表 的 建立 和 维护 。 


1.15.1 创建 和 删除 数据 库 


1. 创建 数据 库 


在 phpMyAdmin 的 管理 界面 中 单 击 “ 数 据 库 ” 超 链接 ， 进 入 新 建 数 据 库 页 面 ， 在 该 页 面 中 有 一 个 
文本 框 、 一 个 下 拉 菜 单 和 一 个 “创建 ”按钮 ， 首 先 在 文本 框 中 输入 数据 库 的 名 称 ， 然 后 选择 编码 ， 最 


局 
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后 单 击 “ 创 建 ” 按 钮 ,这样 新 的 数据 库 就 可 以 被 创建 成 功 . 例如 , 创建 一 个 名 称 为 db_pursey 的 数据 库 ， 
首先 在 文本 框 中 输入 db_pursey， 之 后 在 下 拉 菜 单 中 选择 要 使 用 的 编码 ， 在 Windows 下 一 般 选 择 
utf8_general ci， 如 图 1.46 所 示 。 最 后 单 击 “ 创 建 ”按钮 ， 这 样 名 为 db_pursey 的 数据 库 就 被 创建 成 功 。 
创建 数据 库 后 ， 该 数据 库 名 称 出 现在 左 侧 的 数据 库 列 表 中 ， 单 击 这 个 数据 库 ， 在 右 侧 界面 中 可 以 对 
该 数据 库 进 行 操作 ， 如 结构 、SQL、 导 入 、 操 作 、 权 限 等 ， 单 击 相应 的 按钮 即 可 进入 相应 的 操作 界面 。 


CE > 口 目 


图 1.46 phpMyAdmin 管理 界面 





2. 删除 数据 库 

要 删除 某 个 数据 库 ， 首 先 在 左 侧 的 下 拉 菜单 中 选择 该 数据 库 ， 然 后 单 击 右 侧 界面 中 的 “操作 ” 超 
链接 ， 在 操作 数据 库 的 页 面 中 单 击 “ 删 除数 据 库 ” 超 链接 即 可 。 
1.15.2 ”创建 和 删除 数据 表 

针对 表 级 操作 是 在 选 定 了 数据 库 的 情况 下 进行 的 , 即 表 级 操作 的 前 提 是 用 户 必须 选择 一 个 数据 库 ， 
在 该 数据 库 中 进行 表 的 建立 和 维护 。 

1. 创建 数据 表 


创建 数据 库 db_pursey 后 ， 在 右 侧 页 面 中 会 出 现 如 图 1.47 所 示 的 数据 表 创 建 提示 页 面 ， 完 成 数据 
表 的 创建 操作 。 





图 1.47 数据 表 创建 页 面 


首先 在 表单 中 输入 数据 表 的 名 称 和 字段 数 ， 然 后 单 击 “执行 ”按钮 ， 进 入 各 个 字段 的 详细 信息 录入 表 
单 ， 包 括 字段 名 、 数 据 类 型 、 长 度 / 值 和 默认 值 等 ， 在 这 里 就 完成 了 对 表 结 构 的 详细 设置 ， 如 图 1.48 所 示 。 
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1.48 创建 表 信息 
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当 所 有 的 信息 都 输入 完 以 后 ， 就 可 以 单 击 “ 保 存 ” 按 钮 ， 成 功 创建 数据 表 tb_admin。 一 个 新 的 数 
据 表 被 创建 后 ， 进 入 到 数据 表 页 面 ， 在 这 里 可 以 通过 改变 表 的 结构 来 修改 表 ， 可 以 执行 添加 新 的 列 、 
删除 列 、 修 改 列 的 数据 类 型 或 者 字段 的 长 度 / 值 等 操作 ， 如 图 1.49 所 示 。 





# 名 字 类 型 Ea 后 性 空 又 认 天 外 天 作 


Dl mn 否 天 AVTOINCREMEN 全 各 到 加 这 三 信 DISTINCT) 为 王 和 
修改 字段 属性 删除 字段 3 SC we) Sees DETNCT) 3 
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添加 新 字段 
图 1.49 操作 列表 




















2. 删除 数据 表 


执行 删除 表 的 操作 很 简单 ， 只 需 单 击 右 侧 界面 中 的 “操作 ” 超 链接 ， 在 操作 数据 表 的 页 面 中 单 击 
“删除 数据 表 ” 超 链接 ， 就 可 以 轻松 地 删除 当前 数据 表 。 


1.16 本 章 总 结 


本 章 依据 软件 开发 流程 介绍 了 52 同城 信息 网 的 开发 过 程 。 在 开发 任何 一 个 项 目前 ,首先 要 充分 做 
好 前 期 准备 ， 如 完善 的 需求 分 析 、 清 晰 的 业务 流程 、 合 理 的 程序 结构 等 ， 这 样 在 后 期 的 程序 开发 中 才 
会 得 心 应 手 ， 有 备 无 患 。 通 过 本 章 的 学 习 ， 读 者 可 以 了 解 应 用 phpMyAdmin 创建 数据 库 和 数据 表 的 方 
法 ， 熟 悉 框 架 技术 在 Web 应 用 程序 中 的 应 用 。 
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BCTY365 网 上 社区 


(Apache+PHP+phpMyAdmin+MYySQL 5.5 实现 ) 


随 着 人 类 文明 的 不 断 进步 ， 网 络 这 个 虚拟 世界 也 在 发 生 着 变化 ， 现 
实 世 界 中 的 所 有 内 容 在 网 络 上 都 有 反映 。 现 实 世 界 中 物 以 类 聚 ， 人 以 群 
分 ， 网 络 世界 中 也 形成 了 各 式 各 样 的 社区 。 这 种 社区 建立 在 将 殊 的 兴趣 
点 (如 体育 、 新 闻 、 完 物 、 电 影 和 游戏 等 )、 将 殊 的 人 群 、 将 殊 的 爱好 
或 者 是 将 殊 的 服务 上 等 。 社 区 也 是 一 种 将 定 的 商业 模式 ， 它 本 身 可 以 进 
行 电子 商务 ， 也 可 以 完全 与 商务 无 关 。 

所 谓 网 上 社区 是 指 包 括 BBS/ 论 坛 、 聊 天 室 、 博 容 等 形式 在 内 的 网 上 
交流 空间 ， 同 一 主题 的 网 上 社区 集中 了 有 具有 共同 兴趣 的 访问 者 ， 由 于 有 
众多 用 户 的 参与 ， 因 此 具备 了 交流 的 功能 ， 成 为 一 个 营销 场所 。 

网 上 社区 有 各 种 不 同 的 表现 形式 和 规模 ， 有 个 人 创办 的 社区 ， 功 能 
和 界面 追求 时 尚 、 个 性 突出 ) 有 大 型 的 商业 性 质 社区 ， 以 盈利 为 目的 ， 
分 类 多 元 化 ， 适 合 不 同类 型 的 网 民 。 

本 章 开发 的 BCTY365 网 上 社区 主要 面向 程序 开发 人 员 , 集 论坛 .留言 板 、 
软件 下 载 、 升 级 下 载 、 技 术 支 持 和 在 线 购物 等 功能 于 一 身 ， 既 一 
是 一 个 程序 开发 者 交流 的 平台 ， 更 是 一 个 网 络 营 销 的 场所 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 
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2.1 开发 背景 


随 着 市 场 竞 争 的 日 益 激烈 ， 企 业 的 生存 和 发 展 之 路 更 加 艰难 ， 要 想 使 企业 保持 旺盛 的 生命 力 ， 企 
业 必 须要 跟 上 时 代 发 展 的 脚步 ， 不 断 为 企业 注入 新 的 活力 。 某 科技 公司 为 适应 市 场 的 需求 ， 增 加 公司 
在 互联 网 上 的 影响 力 ， 将 开发 一 个 网 上 社区 系统 ， 为 广大 的 编程 爱好 者 提供 一 个 交流 的 平台 ， 并 且 以 
此 来 推广 该 公司 的 软件 产品 。 


22 系统 分 析 


当 一 个 开发 项 目 被 确立 时 ， 首 先 要 做 的 就 是 需求 分 析 、 可 行 性 分 析 ， 然 后 编写 项 目 计 划 书 ， 以 使 
项 目 开 发 人 员 了 解 和 掌握 网 站 的 前 期 策划 和 网 站 开发 流程 。 


2.2.1 需求 分 析 


在 开发 网 上 社区 之 前 ， 首 先 要 明确 所 要 开发 的 社区 属于 什么 类 型 ， 是 个 人 的 社区 系统 ， 还 是 商业 
化 的 社区 系统 ， 并 且 要 知道 开发 的 社区 是 面向 什么 样 的 人 群 ， 是 普通 网 民 ， 还 是 专业 的 技术 人 员 ， 或 
者 是 其 他 的 特殊 群体 。 针 对 不 同 的 人 群 ， 社 区 应 该 具有 不 同 的 特点 。 当 明确 了 这 些 ， 项 目 开发 的 思路 
就 清晰 了 ， 然 后 再 对 网 站 上 一 些 相关 的 社区 进行 考察 、 分 析 ， 从 中 吸取 经 验 ， 并 结合 企业 的 要 求 以 及 
实际 的 市 场 调查 结果 ， 提 出 一 个 合理 的 网 上 社区 网 站 功能 架构 。 本 网 站 需求 如 下 : 

回 ”网 站 设计 页 面 要求 整 洁 、 美 观 大 方 ， 能 够 展示 企业 形象 。 
网 站 页 面具 有 Banner 广告 ， 树 立 企业 良好 的 口碑 宣传 。 
设计 主要 从 编程 者 的 角度 考虑 ， 为 编程 者 解决 在 开发 中 出 现 的 问题 。 
展示 出 企业 全 力 推 出 的 软件 产品 和 提供 的 免费 软件 ， 以 此 吸引 浏览 者 。 
提供 一 个 良好 的 网 上 购物 的 操作 平台 。 
提供 技术 支持 ， 解 决 编程 过 程 中 常见 的 问题 。 
提供 一 个 讨论 和 研究 问题 的 平台 。 
做 到 让 广大 浏览 者 关注 企业 的 动态 。 
为 客户 提供 反馈 信息 的 平台 ， 能 够 做 到 及 时 与 客户 进行 沟通 。 
完善 的 后 台 管理 系统 。 


2.2.2 可行 性 分 析 


可 行 性 分 析 是 世界 上 普遍 采用 的 一 种 研究 工程 项 目 是 否 可 行 的 科学 。 其 通过 各 种 有 效 的 方法 ， 对 
工程 项 目 进行 分 析 ， 从 技术 、 经 济 、 市 场 等 方面 加 以 评价 ， 最 终 给 投资 决策 者 提供 是 否 选 择 该 项 目 进 
行 开发 的 依据 。 
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BCTY365 网 上 社区 项 目 开发 的 可 行 性 分 析 主 要 从 以 下 两 个 方面 考虑 。 
1. 经 济 可 行 性 分 析 


企业 为 扩大 公司 的 影响 力 ， 推 出 软件 产品 ， 采 用 网 上 社区 的 形式 在 网 络 上 进行 推广 ， 不 但 可 以 汇 
聚 更 多 的 人 气 ， 而 且 可 以 让 更 多 的 人 了 解 该 企业 ， 从 而 达到 推广 企业 软件 产品 的 目的 ， 最 终 为 企业 带 
来 更 大 的 收益 。 更 重要 的 一 点 是 采取 该 方法 的 成 本 相对 其 他 的 电视 广告 或 者 人 力 宣传 的 成 本 要 低 得 多 ， 
虽然 周期 很 长 ， 但 是 却 能 够 取得 长 期 的 收益 。 


2. 技术 可 行 性 分 析 


网 上 社区 系统 的 开发 采用 的 是 Apache+PHP+phpMyAdmin+MySQL 5.0， 开 发 软件 都 是 免费 的 ， 可 
以 直接 从 网 上 下 载 ， 无 须 支付 任何 费用 。 要 完成 BCTY365 网 上 社区 系统 的 开发 ， 必 须 能 够 配置 PHP 
程序 开发 的 环境 ， 掌 握 在 线 支付 、 购 物 车 和 在 线 论坛 技术 。 


2.2.3 ”编写 项 目 计划 书 


根据 《计算 机 软件 产品 开发 文件 编制 指南 》(GB8567 一 1988) 中 的 项 目 开 发 计划 要 求 ， 结 合 单位 
实际 情况 ， 设 计 项 目 计 划 书 如 下 。 


1. 引言 


(1) 编写 目的 。 

为 了 保证 项 目 开发 人 员 按时 保质 地 完成 预订 目标 ， 更 好 地 了 解 项 目 实际 情况 ， 按 照 合理 的 顺序 开 
展 工作 ， 现 以 书面 的 形式 将 项 目 开发 生命 周期 中 的 项 目 任务 范围 、 项 目 团队 组 织 结构 、 团 队 成 员 的 工 
作 责 任 、 团 队 内 外 沟通 协作 方式 、 开 发 进度 、 检 查 项 目 工作 等 内 容 描述 出 来 ， 作 为 项 目 相 关 人 员 之 间 
的 共识 和 约定 ， 作 为 项 目 生命 周期 内 的 所 有 项 目 活动 的 行动 基础 。 

(2) 背景 。 

BCTY365 网 上 社区 系统 是 本 公司 与 X XX 信息 科技 有 限 公 司 签 定 的 待 开发 项 目 ， 网 站 性 质 为 信息 
服务 类 型 ， 为 企业 与 客户 、 浏 览 者 和 会 员 之 间 提供 一 个 技术 交流 平台 ,并且 全 力 推 出 企业 的 软件 产品 。 
项 目 周期 为 两 个 月 ， 项 目 背 景 规划 如 表 2.1 所 示 。 


表 2.1 项 目 背景 规划 









项 目 名 称 签 定 项 目 单位 


甲 方 ，X XX 信息 科技 有 限 公 司 


项 目 负责 人 
甲 方 ; 赵 经 理 







项 目 承担 部 门 
设计 部 门 

开发 部 门 

测试 部 门 











BCTY365 网 上 社区 系统 








乙方 : XXX 网 络 科技 有 限 公 司 乙方 : 张 经 理 


2. 概述 


(1) 项 目 目标 。 
项 目 目标 应 当 符合 SMART 原则 ， 把 项 目 要 完成 的 工作 用 清晰 的 语言 描述 出 来 。BCTY365 网 上 社 


区 系统 的 项 目 目标 如 下 : 
@ 
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BCTY365 网 上 社区 系统 主要 用 于 在 网 络 中 树立 企业 的 形象 ， 为 程序 开发 者 提供 一 个 交流 的 平台 ， 
拉 近 企业 与 客户 、 会 员 和 浏览 者 之 间 的 距离 ， 从 而 达到 推广 企业 软件 产品 的 目的 。 整 个 项 目 需 要 两 个 
月 结束 ， 交 给 客户 进行 验收 。 

(2) 产品 目标 与 范围 。 

一 方面 BCTY365 网 上 社区 系统 能 够 为 企业 节省 大 量 人 力 资源 , 企业 不 再 需要 大 量 的 业务 人 员 去 跑 
市 场 ， 间 接 为 企业 节约 了 成 本 。 另 一 方面 ，BCTY365 网 上 社区 系统 能 够 收集 海量 编程 问题 的 解决 方案 
和 好 的 建议 ， 将 会 有 大 量 用 户 访问 网 站 ， 有 助 于 提高 企业 知名 度 。 

(3) 应 交付 成 果 。 

项 目 开发 完成 后 ， 交 付 的 内 容 如 下 : 

回 ”以 资源 包 的 形式 交付 BCTY365 网 上 社区 系统 的 源 程序 、 网 站 数据 库 文件 、 系 统 使 用 说 明 书 。 

回 客户 方 应 用 自己 的 服务 器 ， 因 此 需要 乙方 架设 Apache 服务 器 ， 安 装 PHP 开发 环境 ， 协 助 甲 

方 购买 域名 ， 将 开发 的 BCTY365 网 上 社区 系统 发 布 到 互联 网 上 运行 。 

回 ”网 站 发 布 到 互联 网 上 以 后 ， 进 行 后 期 的 6 个 月 无 偿 维护 与 服务 ， 超 过 6 个 月 后 进行 网 站 有 偿 

维护 与 服务 。 

(4) 项 目 验收 方式 与 依据 。 

项 目 验收 分 为 内 部 验收 和 外 部 验收 两 种 方式 。 在 项 目 开发 完成 后 ， 首 先进 行内 部 验收 ， 由 系统 测 
试 员 根据 用 户 需求 和 项 目 目标 进行 验收 。 项 目 在 通过 内 部 验收 后 ， 交 给 用 户 进行 验收 ， 验 收 的 主要 依 
据 为 需求 规格 说 明 书 。 


3. 项 目 团队 组 织 


(1) 组 织 结构 。 
为 了 完成 BCTY365 网 上 社区 系统 的 项 目 开 发 ， 公 司 组 建 一 个 临时 的 项 目 团队 ， 由 项 目 经 理 、 系 统 
分 析 师 、PHP 开发 工程 师 、 网 页 设计 师 和 系统 测试 员 构 成 ， 如 图 2.1 所 示 。 





项 目 经 理 


系统 分 析 古 PIP 开 发 工程 师 网 页 设计 师 系统 测试 员 
2.1 项 目 团队 组 织 结构 图 
(2) 人 员 分 工 。 
为 了 明确 项 目 团队 中 每 个 人 的 任务 分 工 ， 现 制定 人 员 分 工 表 ， 如 表 2.2 所 示 。 
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表 2.2 人 员 分 工 表 






























技术 水 平 所 属 部 门 角 色 工作 描述 

负责 项 目的 审批 、 决 策 的 实施 、 项 目的 前 期 分 

张 达 明 | MBA 项 目 开 发 部 | 项 目 经 理 析 、 策 划 、 项 目 开发 进度 的 跟踪 、 项目 质量 的 
检查 

王 言 辉 | 高 级 系统 分 析 师 项 目 开 发 部 | 系统 分 析 师 负责 系统 功能 分 析 、 系 统 框架 设计 

潘 攀 高 级 PHP 工程 师 项 目 开 发 部 | PHP 开发 工程 师 | 负责 软件 前 后 台 设 计 与 编码 

刘 悦 高 级 美工 设计 师 设计 部 网 页 设计 师 负责 网 页 风格 的 确定 、 网 页 图 片 的 设计 

高 级 系统 测试 工程 师 | 项 目 开发 部 | 系统 测试 员 对 软件 进行 测试 、 编 写 软件 测试 文档 








23 系统 设计 


2.3.1 系统 目标 


根据 对 目前 网 络 上 各 种 社区 的 分 析 和 研究 ， 结 合 本 项 目的 实际 需求 ， 在 设计 时 应 该 满足 以 下 目标 ; 
界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵 活 ， 树 立 企业 形象 。 
功能 完善 、 结 构 清晰 。 

重点 突出 企业 的 软件 产品 。 

及 时 更 新 网 站 公告 。 

及 时 查阅 和 回复 客户 反馈 信息 。 

为 用 户 提供 沟通 和 交流 的 平台 。 

购物 车 模块 的 设计 结构 合理 、 流 程 清晰 。 

购物 结算 功能 设计 符合 逻辑 ， 计 算 准确 。 

订单 处 理 功能 的 设计 及 时 、 准 确 、 安 全 。 

处 理 好 网 上 支付 功能 的 设计 与 网 上 银行 之 间 数 据 的 传递 。 

具备 完善 的 后 台 管理 功能 ， 能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 
系统 运行 稳定 ， 具 备 良 好 的 防范 措施 。 


系统 功能 结构 


结合 需求 分 析 和 系统 目标 中 的 内 容 ，BCTY365 网 上 社区 系统 的 功能 结构 已 经 设计 完成 。 为 了 使 读 
者 能 够 更 清楚 地 了 解 网 站 的 结构 ， 下 面 给 出 BCTY365 网 上 社区 前 台 和 后 台 功 能 模块 结构 图 。 

BCTY365 网 上 社区 前 台 管 理 系统 的 功能 设计 如 图 2.2 所 示 。 

BCTY365 网 上 社区 后 台 管 理 系统 的 功能 设计 如 图 2.3 所 示 。 
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论坛 版 块 

查找 帖子 订单 处 理 

帖子 主题 工行 网 上 支付 

帖子 浏览 购物 车 

发 布 帖子 购物 结算 

回复 帖子 

版 块 跳 转 
添加 升级 包 添加 编程 词典 版 本 
编辑 升级 包 编辑 编程 词典 版 本 
添加 序列 号 添加 编程 词典 
tide 编辑 编程 词典 
























































哈 沪 对 误 
哈 沪 模 游 























型 蕊 着 蔡 吉 这 
型 勤 状 将 拱 游 




















图 2.3 网 上 社区 后 台 功 能 模块 结构 图 
2.3.3 ”系统 预览 


BCTY365 网 上 社区 系统 由 多 个 程序 页 面 组 成 ， 下 面 给 出 几 个 典型 页 面 ， 其 他 页 面 参见 资源 包 中 的 
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源 程序 。 
前 台 首页 如 图 2.4 所 示 ， 该 页 面 用 于 展示 本 系统 的 功能 模块 ， 突 出 企业 的 形象 ， 推 广 企业 的 软件 产品 。 
后 台 首 页 如 图 2.5 所 示 ， 该 页 面 用 于 实现 对 编程 词典 、 技 术 支 持 、 软 件 升级 、 软 件 试用 等 内 容 的 管理 。 














图 2.4 前 台 首页 (资源 包 \TM\02\bcty365\index.php) ”图 2.5 ”后台 首页 (资源 包 \TM\02\bcty365\admin\default.php) 


在 线 订 购 模 块 的 页 面 效果 如 图 2.6 所 示 , 该 页 面 主要 用 于 展示 本 企业 在 线 推出 的 软件 产品 , 实现 对 
产品 的 在 线 购买 功能 。 软件 下 载 模块 的 页 面 效 果 如 图 2.7 所 示 , 该 页 面 主要 用 于 展示 本 企业 提供 的 免费 
软件 ， 并 且 提 供 下 载 链接 。 





2.6 ”在线 订购 (资源 包 \TM\02\bcty365\morebccd.php) 图 2.7 软件 下 载 (资源 包 \TM\02\bcty365\rjxz.php) 
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社区 论坛 模块 的 页 面 效 果 如 图 2.8 所 示 , 该 页 面 主要 用 于 展示 论坛 中 的 各 大 版 块 , 并 且 提 供 超 链接 
跳 转 到 对 应 的 版 块 。 后 台 的 登录 页 面 效 果 如 图 2.9 所 示 ， 该 页 面 主要 实现 后 台 管理 员 登 录 。 











图 2.8 社区 论坛 2.9 后 台 登 录 
(资源 包 \TM\02\bcty365\bbs_index.php) (资源 包 \TM\02\bcty365\admin\index.php) 
2.3.4 开发 环境 
在 开发 BCTY365 网 上 社区 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 


1， 服 务 器 端 


操作 系统 : Windows 7/Linux (推荐 )。 
服务 器 : Apache 2.4.18。 

PHP 软件 : PHP 7.0.12。 

数据 库 : MySQL 5.5.47。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-3.5.8。 
开发 工具 : PhpStorm 2016.3。 

浏览 器 : Google Chrome。 

分 辨 率 : 最 佳 效果 为 1680x1050 像素 。 


客户 端 


浏览 器 :Google Chrome。 
分 辩 率 : 最 佳 效果 为 1680x1050 像素 。 


人 8 
9 回国 NN 罗 罗 四 办 罗 罗 四 加 
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2.3.5 ”文件 夹 组 织 结构 


在 进行 网 站 开发 之 前 ， 要 对 网 站 的 整体 文件 夹 组 织 架构 进行 规划 ， 对 网 站 中 使 用 的 文件 进行 合理 
的 分 类 ， 分 别 放置 于 不 同 的 文件 夹 下 。 通 过 对 文件 夹 组 织 架构 的 规划 ， 可 以 确保 网 站 文件 目录 明确 、 
条 理 清 晰 ， 同 样 也 便于 网 站 后 期 的 更 新 和 维护 。 本 案例 的 文件 夹 组 织 结构 如 图 2.10 所 示 。 








Y Dbcty365 
Y 户 admin 

» Dbccdimages 

» ess 

» Dimages 

» Ds 

* 站 sof 

户 conn 














Dess 





户 data 





Dimages 


vvvv 


DD upfile 





网 站 根 目录 

用 于 存储 网 站 后 台 管 理 文件 

用 于 存储 编程 词典 的 界面 

用 于 存储 网 站 后 台 使 用 的 CS5 文件 
用 于 存储 网 站 后 台 的 图 片 文件 

用 于 存储 升级 下 载 的 文件 

用 于 存储 可 下 载 的 软件 

用 于 存储 连接 数据 库 的 文件 

用 于 存储 网 站 前 台 使 用 的 CSS 文件 
用 于 存储 数据 库 文件 

用 于 存储 网 站 前 台 使 用 的 的 图 片 文件 
用 于 存储 网 站 论坛 中 上 传 的 图 片 文件 





2.10 文件 夹 组 织 结构 


2.4 在 Linux 操作 系统 下 搭建 PHP 开发 环境 


Red Hat Linux 9 是 Linux 众多 版 本 中 比较 大 众 化 的 一 版 。 在 安装 系统 时 ， 如 果 选 择 完 全 安装 或 者 选 
择 Apache、MySQL、PHP 的 安装 包 ， 则 三 者 将 被 安装 到 系统 中 ， 用 户 只 需 将 Apache 和 MySQL 服务 启 
动 就 可 以 使 用 二 者 ， 非 常 方便 ， 但 是 Apache 和 MySQL 的 版 本 可 能 不 是 很 理想 。 为 了 能 够 创建 一 个 良好 
的 PHP 开发 环境 ， 这 里 将 详细 介绍 自行 在 Linux 下 安装 和 配置 Apache 2+MySQL 5.0+PHP 5 的 方法 。 
首先 应 该 到 相关 官方 网 站 下 载 三 者 的 安装 包 和 libxml2 的 安装 包 : 


网 回回 网 


以 不 必 下 载 该 安装 包 )。 
2.4.1 Linux 下 Apache 的 安装 配置 


首先 将 下 载 的 httpd 安装 包 复 制 到 适当 的 位 置 ， 例 如 
/usr/local/work 下 《如 果 目 录 不 存在 ， 可 以 建立 该 目录 )。 
打开 Red Hat Linux 9 的 主 菜单 ， 选 择 “ 系 统 工具 ”， 在 弹 
出 的 菜单 中 选择 “终端 ”命令 ,将 打开 如 图 2.11 所 示 的 终 
端 窗口 。 


httpd-2.0.58.tar.gz 或 更 高 版 本 http://httpd.apache. 
mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 或 更 高 版 本 (http://www.mysql.comy/)。 
php-5.0.0.tar.gz 或 更 高 版 本 (http://www.php.net/)。 

libxml2-2.5.10.tar.gz 或 更 高 版 本 (如 果 读 者 系统 中 libxml2 的 版 本 已 经 等 于 或 高 于 该 版 本 ， 可 


org/)。 
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图 2.11 Red Hat Linux 9 的 终端 命令 窗口 


@ 
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AS 四 Linux 下 Apache、MySQL 及 PHP 的 安装 都 是 在 如 图 2.11 所 示 的 终端 命令 窗口 中 通过 命 
令 方 式 实现 的 。 


在 该 窗口 中 输入 如 下 命令 进入 work 目录 : 


cd/usr/local/work 





在 work 目录 中 输入 如 下 命令 解压 httpd-2.0.58.tar gz: 

tar xfz httpd-2.0.58.tar.gz 

进入 解压 后 的 目录 httpd-2.0.58: 

cd httpd-2.0.58 

建立 makefile， 并 将 Apache 2 安装 到 /ust/local/apache2 目录 下 : 
./configure —prefix=/usr/local/apache -enable-module=so 

开始 编译 : 

make 

开始 安装 到 设置 的 目录 中 : 

make install 

到 此 Apache 2 的 安装 工作 完成 , 可 以 在 每 次 启动 系统 时 通过 如 下 命令 启动 或 重新 启动 Apache 2 服务 : 


/usr/local/lapache2/bin/apachectl start 
/usr/local/apache2/bin/apachectl restart 





打开 浏览 器 ， 在 地 址 栏 中 输入 http://127.0.0.1 或 者 http://localhost， 按 Enter 键 ， 如 果 出 现 如 图 2.12 
所 示 的 页 面 ， 则 说 明 Apache 2 安装 成 功 。 
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Vs 一 = , 
千本 本 


PY CE Ee ES 
图 2.12 测试 Apache 服务 器 






























































第 2 章 BCTY365 网 上 社区 ( Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


2.4.2 Linux 下 MySQL 的 安装 配置 


将 mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 复制 到 /usr/local/work 目录 下 ， 建 立 MySQL 账号 : 
groupadd mysql 


在 组 群 中 加 入 MySQL: 





useradd --g mysql mysql 


进入 local 目录 : 
cd/usr/local 


将 mysql-standard-5.0.0-alpha-pc-linux-i686.targz 解压 到 该 目录 : 





tar xfz/usr/local/work/mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 


考虑 到 MySQL 数据 库 升级 的 需要 ， 所 以 通常 以 链接 的 方式 建立 usrlocalmysql 目录 ; 





In -s mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz mysql 





进入 MySQL 目录 : 

cd mysql 

在 /ust/local/mysql/data 中 建立 MySQL 的 数据 库 : 
scripts/mysql_install_db -user=mysql 

修改 文件 权限 : 


chown -R root . 
chown -R mysql data 
chgrp -R mysql . 








到 此 MySQL 安装 成 功 。 可 以 通过 在 终端 中 输入 如 下 命令 启动 MySQL 服务 : 


/usr/local/mysql/bin/mysqld_safe -user=mysql & 


启动 MySQL 后 输入 如 下 命令 查看 安装 结果 : 


/usr/local/mysql/bin/mysql —uroot 


如 果 终 端 窗口 出 现 如 图 2.13 所 示 的 提示 ， 则 说 明 MySQL 安装 成 功 。 
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图 2.13 测试 MySQL 
2.4.3 Linux 下 PHP 的 安装 配置 


首先 查看 系统 中 libxml2 的 版 本 号 ， 如 果 libxml2 的 版 本 号 小 于 2.5.10， 则 需要 安装 libxml2- 
2.5.10.tar.gz 或 更 高 版 本 ， 因 为 PHP 5 必须 在 libxml2 的 版 本 大 于 2.5.10 的 前 提 下 才能 够 安装 。 
将 libxml2-2.5.10.tar.gz 复制 到 /usrlocal/work 目录 下 ， 并 进入 该 目录 : 


cd/usr/local/work 

解压 libxml2-2.5.10.tar.gz: 
tar xfz libxml2-2.5.10.tar.gz 
进入 该 目录 : 


cd libxml2-2.10 





建立 makefile 并 将 libxml2 安装 到 /srlocallibxml2 下 : 
./configure —prefix=/usr/local/libxml2 
开始 编译 : 


make 








开始 安装 到 设置 的 目录 去 : 


make install 


到 此 libxml2 安装 成 功 。 
将 php-5.0.0.tar.gz 复制 到 /usr/local/work 目录 下 ， 并 进入 该 目录 : 


cd/usr/local/work 
解压 php-5.0.0.tar.gz: 
tar xfz php-5.0.0.tar.gz 


进入 php-5.0.0 目录 : 





cd php-5.0.0 


@_ 
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建立 makefile: 
./configure -with-apxs2=/usrlocal/apache2/bin/apxs\ 


—with-mysql=/usr/local/mysql \ 
—wWith-libxml-dir=/usr/local/libxml2 


开始 编译 : 
make 

开始 安装 : 
make install 


复制 php.ini-dist 或 php.ini-recommended 到 /usr/local/lib 目录 ， 并 命名 为 php.ini: 





cp php.ini-dist/usr/local/lib/php.ini 


更 改 httpd.conf 文件 相关 设置 , 该 文件 位 于 /usr/local/apache2/conf 中 。 找 到 该 文件 中 的 如 下 指令 行 : 





AddType application/x-gzip .gz .tgz 
在 该 指令 后 添加 如 下 指令 : 

AddType application/x-httpd-php .php .phtml 

重新 启动 Apache， 并 在 Apache 主 目录 下 建立 文件 test.php: 


<?php 
phpinfo(); 
?> 


在 浏览 器 中 输入 http://127.0.0.1/test.php， 按 Enter 键 ， 如 果 出 现 如 图 2.14 所 示 的 页 面 ， 则 PHP 安 
装 成 功 。 
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图 2.14 PHP 测试 
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od 
ME 培 明 Apache2 默认 主 目 录 为 /usr/local/apache2/htdocs。 


二 


[Ea 安装 文件 的 路 径 要 遵循 一 定 的 客观 原则 ， 为 了 避免 在 Windows 和 Linux 间 移植 程 序 时 带 
来 的 不 便 ， 选 择 D:\usr\local\php 的 目录 时 要 和 在 Linux 下 的 安装 目录 相 匹配 。 建 议 最 好 不 要 选择 中 
间 有 空格 的 目录 ， 如 E:\Program Files\PHP， 这 样 做 会 导致 一 些 未 知 错误 或 崩 演 发 生 。 





2.5 数据 库 设 计 





视频 讲解 “| 


开发 一 个 功能 完善 的 网 上 社区 离 不 开 数 据 库 的 支持 ， 只 有 拥有 了 强大 的 数据 库 ， 网 上 社区 才能 够 
存储 大 量 的 数据 信息 ,实现 更 多 、 更 好 的 功能 来 吸引 更 多 的 社区 成 员 。 本 节 将 对 BCTY365 网 上 社区 数 
据 库 的 设计 进行 详细 介绍 。 


2.5.1 数据库 分 析 


BCTY365 网 上 社区 是 一 个 中 型 的 面向 软件 开发 者 的 程序 ， 考 虑 到 开发 的 成 本 、 搭 配 的 合理 性 以 及 
操作 的 灵活 性 等 ， 使 用 了 MySQL 数据 库 。 从 成 本 考虑 MySQL 数据 库 是 完全 免费 的 ， 可 以 在 网 上 免费 
下 载 ， 从 匹配 的 角度 讲 ，PHP 与 MySQL 数据 库 一 直 是 公认 的 最 佳 搭档 ; MySQL 数据 库 不 但 可 以 在 命 
令 模 式 下 进行 操作 ， 而 且 还 配备 了 一 些 比较 流行 的 图 形 化 管理 工具 ， 如 phpMyAdmin 等 ， 可 以 轻松 地 
对 MySQL 数据 库 中 的 数据 进行 操作 。 


2.5.2 数据库 概念 设计 


根据 上 述 各 节 对 BCTY365 网 上 社区 系统 做 的 需求 分 析 和 系统 设计 ， 规 划 出 BCTY365 网 上 社区 的 
实体 关系 E-R 图 。 其 中 包括 注册 用 户 实体 、 发 帖 信息 实体 、 回 帖 信息 实体 、 订 单 信息 实体 和 编程 词典 
信息 实体 ， 其 他 还 有 一 些 辅助 的 实体 ， 是 对 上 述 实体 中 内 容 的 补充 。 由 于 涉及 的 实体 较 多 ， 这 里 只 对 
注册 用 户 实体 、 发 帖 信息 实体 和 订单 信息 实体 的 E-R 图 进行 介绍 。 


1. 注册 用 户 实体 


注册 用 户 实体 用 于 存储 用 户 注册 信息 , 包括 
编号 、 用 户 名 、 真 实 姓名 、 密 码 、 邮 箱 、 性 别 、 
电话 、QQ 号 码 、 家 庭 地 址 、 访 问 次 数 、 注 册 时 
间 、 最 后 一 次 登录 时 间 、IP 地 址 、 邮 政 编码 、 
用 户 类 型 、 密 码 提 示 问 题 、 密 码 答 案 、 真 实 密码 、 
表情 图 、 发 帖 次 数 等 属性 。 注 册 用 户 实体 的 E-R 
图 如 图 2.15 所 示 。 


局 





图 2.15 注册 用 户 实体 E-R 图 


第 2 章 BCTY365 网 上 社区 (Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


2. 发 帖 信息 实体 


发 帖 信息 实体 用 于 存储 登录 本 社区 的 会 员 在 论坛 中 发 布 帖子 的 相关 信息 ， 包 括 编号 、 用 户 名 外 、 
帖子 类 型 、 帖 子 标题 、 帖 子 内 容 、 发 帖 时 间 、 最 后 回复 时 间 、 表 情 图 、 访 问 次 数 、 是 否 项 帖 、 上 传 图 
片 等 属性 。 发 帖 信息 实体 的 E-R 图 如 图 2.16 所 示 。 














3. 订单 信息 实体 


订单 信息 实体 存储 用 户 在 线 购买 时 填写 的 订单 信息 ， 包 括 编号 、 用 户 名 、 性 别 、 家 庭 地 址 、 邮 政 
编码 、QQ 号 码 、 邮 箱 、 手 机 号 码 、 电 话 号 码 、 收 货 方式 、 邮 资 、 产 品 金额 、 订 单 时 间 、 订 单 号 、 选 择 
城市 等 属性 。 订 单 信息 实体 的 E-R 图 如 图 2.17 所 示 。 





2.16 发 帖 信息 实体 E-R 图 2.17 订单 信息 实体 E-R 图 
2.5.3 创建 数据 库 及 数据 表 


在 BCTY365 网 上 社区 系统 中 应 用 的 是 db_bcty365 数据 库 ， 其 中 涉及 18 个 数据 表 ， 数 据 表 的 名 称 
和 功能 如 表 2.3 所 示 。 


表 2.3 db_bcty365 数据 库 中 的 数据 表 



































表 说 阴 
tb bb 编程 词典 版 本 信息 表 
tb bbqb utfg_general ci 版 本 之 间 区 别 信息 表 
tb bbs utfg_general ci 论坛 发 帖 信息 表 
tb bccd utf8_general ci 编程 词典 信息 表 
tb _city utf8_general ci 城市 信息 表 
tb_cjwt utf8_general ci 常见 问题 信息 表 
tb dd utf8 general ci 订单 信息 表 
tb reply ntf8_general ci 论坛 回帖 信息 表 
tb sjxz utf8 general ci 软件 升级 下 载 信息 表 
tb_soft utf8 general ci 软件 下 载 信息 表 
tb tell utf8 general ci 社区 公告 信息 表 
tb type utf8 general ci 社区 模块 类 型 信息 表 
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续 表 

表 说 明 
tb type_ big 论坛 大 类 信息 表 
tb type small 论坛 小 类 信息 表 
tb user utf8 general ci 注册 用 户 信息 表 
tb xlh utf8 general ci 升级 下 载 序 列 号 信息 表 
tb_bccdjj utf8 general ci 编程 词典 简介 信息 表 
tb leaveword utf8 general ci 存储 客户 反馈 信息 





本 案例 中 创建 数据 库 和 数据 表 使 用 的 是 phppMyAdmin 图 形 化 管理 工具 ， 下 面 将 介绍 数据 库 和 数据 
表 的 创建 方法 ， 以 及 在 创建 过 程 中 需要 注意 的 一 些 问题 。 


1. 数据 库 的 创建 


打开 phpMyAdmin 图 形 化 管理 工具 的 主页 ， 首 先 在 文本 框 中 输入 要 创建 的 数据 库 的 名 称 〈 如 db_ 
bety365)， 然 后 在 下 拉 列 表 框 中 选择 要 使 用 的 字符 编码 格式 ， 这 里 使 用 的 是 utf8_general ci， 如 图 2.18 
所 示 。 最 后 单 击 “创建 ”按钮 ， 数 据 库 创建 成 功 。 


[人 技巧 创建 数据 库 的 过 程 中, 尽量 使 用 与 程序 内 容 贴切 的 英文 字符 进行 命名 , 有 助 于 对 数据 库 

的 理解 。 如 果 使 用 phpStudy 配置 PHP 开发 环境 ， 那 么 在 使 用 phpMyAdmin 创建 数据 库 时 不 需要 
指定 编码 的 格式 ， 默 认 值 为 utf8_general_ ci; 如 果 自 行 配置 开发 环境 ， 那 么 就 要 指定 编码 格式 为 
utf8_general_ ci， 否则 创建 数据 库 的 编码 格式 为 latin1 _swedish ci， 将 导致 数据 库 中 的 数据 出 现 
乱码 。 


2. 创建 数据 表 


在 成 功 创建 数据 库 以 后 ， 接 下 来 就 是 创建 数据 表 ， 这 里 以 tb_bb 编程 词典 版 本 信息 表 为 例 ， 讲 解 
如 何 创建 数据 表 ， 以 及 在 创建 数据 表 的 过 程 中 都 需要 注意 哪些 问题 。 这 里 创建 一 个 名 为 tb_bb 的 数据 
表 ， 包 括 3 个 字段 ， 如 图 2.19 所 示 。 


数据 库 襄 SQL 局 状态 中 用 户 加 呈 局 导入 少 设 置 











数据 库 选择 编码 格式 



































输入 数据 库 名 翻新 建 数 据 去 
了 库 本 
司 廿 名字 :([tb_bb J |[ 失 和 新 表 名 和 
图 2.18 phpMyAdmin 管理 界面 2.19 创建 由 bb 数据 表 


单 击 “ 执 行 ”按钮 后 ， 进 入 到 如 图 2.20 所 示 的 添加 字段 信息 的 页 面 中 ， 在 此 处 对 字段 进行 详细 的 
设置 ， 包 括 字段 名 、 数 据 类 型 、 长 度 / 值 、 属 性 、 默 认 值 、 额 外 、 主 键 和 索引 等 。 


局 
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BF 关于 马 长 乓 /全 过 于 名 E33 三 宝 素 引 AI 
INT 加 加 口 (人 加 

VARCHAR 指定 数据 表 的 类 型 2 oo i 

指定 编码 格式 0 上 


设置 主键 




















图 2.20 添加 数据 表 中 字段 信息 


| 


| 
序 后 期 维护 和 修改 工作 的 进行 ， 能 够 直观 地 看 出 数据 表 的 作用 。 

如 果 使 用 phpStudy 配置 PHP 开发 环境 , 那么 在 创建 数据 表 时 不 需要 指定 数据 表 类 型 和 编 
码 格式 ; 如 果 使 用 自行 配置 的 PHP 开发 环境 ， 那么 就 要 指定 数据 表 的 类 型 为 MyISAM 和 字符 的 编 
码 格式 为 utf8_general_ci， 否 则 创建 的 数据 表 类 型 为 hnoDB， 而 编码 格式 为 latin1 swedish_ ci， 将 
导致 该 数据 表 中 的 数据 复制 到 其 他 机 器 上 后 不 可 用 ， 并 且 数 据 表 中 的 数据 出 现 乱码 。 

在 创建 数据 表 的 过 程 中 ， 一 定 要 为 数据 表 指 定 一 个 主键 ， 它 是 数据 表 的 唯一 标识 。 


掌握 数据 表 的 创建 方法 后 ， 就 可 以 自行 创建 数据 表 。 由 于 本 案例 中 涉及 的 数据 表 多 达 18 个 ， 这 里 
不 能 对 每 个 数据 表 的 功能 设计 进行 一 一 介绍 ， 所 以 只 给 出 几 个 重要 的 数据 表 的 结构 供 广大 读者 参考 ， 
其 他 数据 表 请 参见 本 书 附 带 的 资源 包 。 数 据 表 的 设计 结构 如 表 2.4 一 表 2.6 所 示 。 
(1) tb_user (注册 用 户 信息 表 )。 
注册 用 户 信息 表 主 要 用 于 存储 本 社区 中 会 员 的 个 人 信息 。 该 数据 表 的 结构 如 表 2.4 所 示 。 


表 2.4 注册 用 户 信息 表 结 构 





























名 字 | 类 型 | 整 理 | 空 | 默 认 | 额 外 | 
记 intg) | 
usernc varchar(50) utf8 general ci | 
truename varchar(50) utf8 general ci 是 NULL 
pwd varchar(50) utf8 general ci NULL 
email varchar(50) utf8 general ci 是 NULL 
SeX varchar(2) utf8_general ci 是 NULL 
tell varchar(20) utf8 general ci 是 NULL 
qq varchar(20) utf8_general ci 是 NULL 
address varchar(100) utf8_general ci 是 NULL 
logintimes int(8) 否 无 
Tegtime datetime 否 无 
lastlogintime | datetime 否 无 
ip varchar(20) utf8 general ci 否 无 
yb varchar(20) utf8 general ci 是 NULL 
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整 理 空 | 默 认 
usertype | int(?) 否 | 无 用 户 类 型 
question | varchar(200) utf8 general ci 否 无 密码 提示 问题 
answer | varchar(200) utf8_general ci 否 无 密码 提示 答案 
truepwd | varchar(200) utf8 general ci 否 无 真实 密码 
photo | varchar(50) utf8 general ci 否 无 表情 图 
pubtimes 是 0 发 帖 次 数 


(2) tb_reply〈 论 坛 回帖 信息 表 )。 
论坛 回帖 信息 表 主要 用 于 存储 登录 会 员 在 本 社区 中 回复 帖子 的 信息 。 该 数据 表 的 结构 如 表 2.5 所 示 。 


表 2.5 论坛 回帖 信息 表 结构 




















名 字 类 型 整理 
id int(8) 

Userid int(8) 

bbsid int(8) 

title varchar(200) utf8 general ci 
content mediumtext utf8 general ci 
createtime “| datetime 

mark int(2) 

photo varchar(80) utf8 general ci 








(3) tb_bccd 编程 词典 信息 表 )。 
编程 词典 信息 表 主 要 用 于 存储 本 社区 的 在 线 订购 模块 中 出 售 的 编程 词典 的 基本 信息 。 该 数据 表 的 
结构 如 表 2.6 所 示 。 


表 2.6 编程 词典 信息 表 结构 









































名 字 类 型 整 理 空 说 阴 
id int(8) 否 | 无 自动 编号 id 
bccdname varchar(200) utf8 general ci 否 无 编程 词典 名 称 
Owner varchar(100) utf8_general ci 否 无 开发 者 
typeid varchar(50) utf8 general ci 否 无 版 本 类 型 
content mediumtext utf8_general ci 否 无 编程 词典 简介 
samepart mediumtext utf8 general ci 否 无 软件 共同 点 
addtime datetime 否 无 开发 时 间 
imageaddress | varchar(100) utf8 general ci 否 无 界面 存储 地 址 
bbid int(8) 是 NULL 所 属 版 本 id 
price float 是 NULL 价格 
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2.6 公共 模块 设计 














2.6.1 ”数据 库 连接 文件 


在 进行 程序 开发 的 过 程 中 ， 有 很 多 地 方 都 涉及 数据 库 的 应 用 ， 在 应 用 数据 库 之 前 首先 要 与 数据 库 
建立 连接 ， 因 此 可 以 将 数据 库 的 连接 代码 作为 一 个 公共 文件 进行 存储 ， 在 需要 使 用 数据 库 连接 文件 的 
地 方 直接 调用 该 文件 即 可 ， 无 须 重 复 编写 相同 的 代码 ， 既 减少 了 代码 的 元 余 ， 也 便于 对 数据 库 连接 文 
件 进 行 修改 。 在 本 项 目 中 笔者 将 数据 库 的 连接 代码 存储 于 conn.php 中 。conn.php 文件 的 代码 如 下 : 


倒 程 01 代码 位 置 ; 资源 包 \TM\02\bety365\conn\conn.php 


<?php 
@ ”$conn=mysqli_connect("localhost","root","root"); // 连 接 数 据 库 服务 器 





© mysqli_select_db($conn,"db_bcty365"); /| 连接 指定 的 数据 库 
© mysqliquery($conn,"set names utf8"); // 对 数据 库 中 的 编码 格式 进行 转换 ， 避 免 出 现 中 文 乱码 的 问题 
?> 

外 代码 贴 二 


@ mysqli_connect0: 连接 MySQL 服务 器 ， 服 务 器 的 用 户 名 为 root， 密 码 为 Toot。 
@ mysqli_select_ db0: 用 于 连接 指定 的 MySQL 数据 库 ， 数 据 库 名 为 db_bcty365。 
@@ set names utf8: 指定 数据 库 中 字符 的 编码 格式 为 utf8。 


成 功 创建 conn.php 文件 后 , 在 需要 进行 数据 库 操作 的 程序 中 , 就 可 以 通过 include 或 者 其 他 包含 语 
旬 调 用 conn.php 文件 即 可 ， 无 须 再 编写 连接 数据 库 的 程序 代码 。 应 用 include 语句 包含 conn.php 文件 
的 代码 如 下 : 


<?php 
include ("conn/conn.php"); // 包 含 数 据 库 文 件 





?> 


2.6.2 ”将 文本 中 的 字符 转换 为 HTML 标识 符 


在 输出 数据 库 中 数据 的 过 程 中 ， 有 必要 将 数据 中 的 一 些 特殊 字符 转换 为 HTML 标识 符 ， 这 样 可 以 
避免 一 些 不 必要 的 麻烦 。 例 如 ， 在 输出 一 个 程序 的 执行 代码 的 过 程 中 ， 如 果 不 对 其 进行 转换 ， 那 么 输 
出 的 将 不 是 程序 的 代码 ， 而 是 程序 的 执行 结果 。 这 里 将 文本 中 字符 的 转换 编写 到 一 个 自 定义 函数 
unhtml0 中 ， 并 保存 到 function.php 文件 中 ， 将 其 作为 一 个 公共 模块 来 使 用 ， 当 需要 使 用 时 直接 调用 
function.php 文件 即 可 。function.php 文件 包含 两 个 自 定义 函数 , 即 unhtml0 函 数 和 msubstr0 函 数 ,unhtml0 
函数 用 于 将 数据 中 的 特殊 字符 转换 为 HTML 标识 符 ; msubstr0 函 数 用 于 对 字符 串 进行 指定 长 度 的 截取 。 


代码 如 下 : 
OD) 
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倒 程 02 代码 位 置 资源 包 \TM\02\bcty365\function.php 





<?php 

function unhtml($content)f /定义 自 定义 函数 的 名 称 

@ S$content=htmlspecialchars($content); /| 转换 文本 中 的 特殊 字符 

@ S$content=str_replace(chr(13),"<br>",$content); // 蔡 换文 本 中 的 换行 符 
S$content=str_replace(chr(32),"&nbsp:", $content); // 蔡 换文 本 中 的 “&nbsp;” 
S$content=str_replace("L[","<",$content]; // 营 换文 本 中 的 大 于 号 
S$content=str_replace(")_)",">",$content); // 营 换文 本 中 的 小 于 号 
S$content=str_replace("|_|"," ",$content); // 营 换文 本 中 的 空格 

目 return trim($content); /删除 文本 中 首尾 的 空格 


} 
// 定 义 一 个 用 于 截取 一 段 字 符 串 的 函数 msubstr() 
function msubstr($str$start,$len){ ” //$str 指 的 是 字符 串 ，$start 指 的 是 字符 串 的 起 始 位 置 ，$len 指 的 是 长 度 


$strlen=$start+S$len; /用 $strlen 存储 字符 串 的 总 长 度 〈 从 字符 串 的 起 始 位 置 到 字符 串 的 总 长 度 ) 
$tmpstr=”; // 对 字符 串 变量 进行 初始 化 赋值 
for($i=0;$i<$strlen, $i++}{ // 通 过 for 循环 语句 ， 循 环 读 取 字符 串 


@ iford(substr($str$i,1))>0xa0){ // 如 果 字 符 串 中 首 个 字 节 的 ASCII 序数 值 大 于 0xa0， 则 表示 为 汉字 
S$tmpstr.=substr($srt,$i,3); /| 每 次 取出 3 位 字符 赋 给 变量 $tmpstr， 即 等 于 一 个 汉字 





$i+=2; /变量 自 加 2 
jelsef // 如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 $tmpstr 
Stmpstr.=substr($str,$i,1);} 
} 
return $tmpstr; // 输 出 字符 串 
}?> 
< 从 代码 贴 十 


@ htmlspecialchars0: 将 特殊 字符 转换 成 HTML 格式 ， 而 不 会 将 所 有 字符 都 转换 成 HTML 格式 。 

@ str replace(mixed search.mixed replace.mixed subject.int &count): 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 
替换 ， 参 数 &count 表示 替换 字符 串 执行 的 次 数 。 
@ trim0: 删除 字符 囊 中 首尾 的 空白 或 者 其 他 字符 。 
@ substr0: 从 指定 的 字符 串 中 按照 指定 的 位 置 截取 一 定 长 度 的 字符 。 





2.7 前 台 首 页 设计 





i 视频 讲解 


当今 时 代 , 很 多 人 都 十 分 重视 事物 的 第 一 印象 ， 第 一 印象 基本 上 就 决定 对 某 个 事物 的 看 法 和 态度 ， 
在 网 络 中 更 是 如 此 ， 网 站 给 人 的 第 一 印象 如 果 不 好 ， 那 么 就 会 有 很 多 人 因此 而 不 去 浏览 该 网 站 ， 无 论 
网 站 的 内 容 是 否 丰富 。 可 以 说 网 站 首页 设计 的 成 功 与 否 直接 影响 着 整个 网 站 的 发 展 。 


2.7.1 前 台 首 页 概述 
网 站 首页 是 整个 网 站 的 脸面 ， 既 要 突出 企业 的 形象 ， 又 要 展示 出 网 站 强大 的 功能 。 如 果 网 站 首页 


设计 得 非常 成 功 ， 那 无 疑 是 为 整个 网 站 的 成 功 增添 了 一 个 硅 码 。BCTY365 网 上 社区 首页 的 设计 以 企业 
的 品牌 形象 为 基础 ， 全 力 打造 网 站 的 整体 功能 ， 重 点 推出 企业 的 软件 产品 ， 具 体内 容 如 下 。 


名 
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加 


网 站 菜单 导航 : 包括 “首页 ”“ 技 术 支 持 ”“ 在 线 订 购 ”“ 社 区 论坛 " “软件 下 载 "”“ 升 级 下 载 ” 
“购买 须知 ”“ 联 系 我 们 ”等 导航 菜单 。 

用 户 注册 和 登录 模块 : 实现 用 户 注册 、 会 员 登 录 、 找 回 密码 和 修改 密码 的 功能 。 

网 站 公告 : 主要 用 于 发 布 社区 中 的 一 些 新 消息 和 重大 事件 。 

编程 词典 模块 : 推广 企业 的 软件 产品 。 

软件 下 载 模块 : 展示 企业 提供 的 适用 版 和 免费 的 软件 产品 。 

常见 问题 模块 : 列举 出 编程 中 常见 问题 的 解决 方案 。 

社区 论坛 模块 : 浏览 社区 论坛 中 的 部 分 帖子 。 

升级 下 载 模块 : 提供 一 些 软件 的 升级 版 本 下 载 。 

上 述 内容 就 是 BCTY365 网 上 社区 首页 中 体现 出 的 内 容 ， 为 了 更 加 直观 地 了 解 网 上 社区 首页 的 设 
计 ， 这 里 先 预览 一 下 社区 首页 ， 该 首页 在 本 书 资源 包 中 的 路 径 为 \TM\02\bcty365\index.php， 如 图 2.21 
所 示 。 

BCTY365 网 上 社区 首页 的 设计 看 上 去 很 复杂 ， 由 多 个 版 块 组 成 ， 但 实现 的 过 程 非常 简单 。 总 体 架 
构 使 用 一 个 2 行 3 列 的 表格 和 一 个 3 行 3 列 的 表格 ， 将 其 分 割 成 不 同 的 版 块 ， 然 后 使 用 脚本 语句 从 数 
据 库 中 读 取 数据 ， 最 后 将 数据 循环 输出 到 页 面 中 ， 其 中 网 站 的 头 尾 文件 使 用 include 包含 语句 调用 。 首 
页 的 框架 结构 如 图 2.22 所 示 。 
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图 2.21 BCTY365 网 上 社区 首页 2.22 ”网 站 首页 的 框架 结构 





2.7.2 ”前 台 首 页 技术 分 析 


作为 网 站 首页 不 一 定 要 具有 什么 特殊 的 技术 或 者 功能 ， 应 该 是 以 简洁 、 鲜 明 、 突 出 企业 形象 、 
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展示 网 站 的 功能 为 主 。 即 使 使 用 的 是 一 个 静态 页 面 ， 只 要 能 够 将 内 容 表 达 全 面 、 完 整 ， 那 么 这 个 首页 
设计 也 是 非常 成 功 的 。 

在 本 案例 首页 的 设计 中 ， 应 用 到 一 个 文字 循环 滚动 的 技术 ， 通 过 该 技术 来 输出 社区 中 发 布 的 公告 
信息 。 该 技术 是 通过 JavaScript 脚本 和 div 标签 来 共同 实现 的 ， 其 实现 的 原理 是 : 首先 创建 一 个 div 标 
签 ， 然 后 在 div 标签 中 输出 公告 信息 ， 最 后 通过 JavaScript 来 对 div 标签 进行 操作 ， 实 现 div 标签 中 内 
容 的 滚动 输出 。 该 技术 的 实现 在 index.php 页 中 完成 ， 其 中 使 用 的 JavaScript 脚本 的 代码 如 下 : 


倒 程 03 代码 位 置 ， 资源 包 \TM\02\becty365\index.php 
<script language="JavaScript"> 


marqueesHeight=222; // 定 义 输出 标签 的 高 度 
stopscroll=false; /定义 stopscroll 的 默认 值 为 false 
with(marqueesjf{ /编辑 marquees 标签 的 属性 
style.height=marqueesHeight; /定义 marquees 标签 的 高 为 222 
style.overflowX="visible"'; /定义 值 为 显示 
style.overflowY="hidden"; // 定 义 值 为 隐藏 
noWrap=true; 


onmouseover=new Function("stopscroll=true"); // 当 鼠标 经 过 时 执行 stopscroll=true 
onmouseout=new Function("stopscroll=false"); // 当 鼠标 离开 时 执行 stopscroll=false 


} 
// 创 建 一 个 新 的 div"templayer" 与 div"marquees" 进 行 连接 ， 实 现 不 间断 的 循环 输出 内 容 
document.write(<div id="templayer style="position:absolute;z-index:1;,visibility:hidden"></div>"); 
preTop=0; currentTop=0; 
function init(){ 
templayer.innerHTML="™"; // 设 置 templayer 的 初始 值 为 空 
while(templayeroffsetHeight<marqueesHeightj{ 。“”// 判 断 当 templayer 的 高 度 小 于 marquees 的 高 度 时 
templayer.innerHTML+=marquees.innerHTML; // 将 templayer 的 值 赋 给 marquees 


marquees.innerHTML=templayer.innerHTML+templayer.innerHTML;// 将 templayer 的 值 累加 
setlnterval("scrollup()",50); /间隔 50 毫秒 执行 一 次 scrollup() 函 数 

} 

function scrollup()f /实现 滚动 输出 
if(stopscroll==true) return; 1/ 判断 如 果 stopscroll==true， 不 执行 循环 
preTop=marquees.scrollTop; 


marquees.scrollTop+=1; 

if(preTop==marquees.scrollTopX{ 
marquees.scrollTop=templayer.offsetHeight-marqueesHeight; 
marquees.scrollTop+=1; 


上 


</script> 


在 div 标签 中 ， 主 要 是 输出 数据 库 中 存储 的 公告 信息 ， 并 且 对 输出 的 信息 进行 截取 和 车 换 ， 规 范 
输出 的 内 容 。div 标签 中 的 程序 代码 如 下 : 


<div id="marquees" class="middle"><!-- 创 建 一 个 div 标签 --> 


<?php // 从 数据 库 中 读 取 公告 数据 
@ 
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S$sql=mysqli_query($conn,"select id ,title,createtime from tb_tell order by createtime desc limit 


0,10"); 
Sinfo=mysqli_fetch_array($sq)); 
if($info==falseX{ /| 判断 当 $info==false 时 执行 下 面 的 内 容 
?> 
<div align="center" style="height:25px:"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div> 
<?php 
Jelse{ 
$i=1; // 定 义 变量 $i=1 
dof /执行 do…while 循环 语句 
?> 
<div class="scroll"> 
<a href="tellinfo.php?id=<?php echo $info[id];?>" class="a1"> 
<?php 
if($i==1X{ // 判 断 当 $i==1 时 ， 将 输出 的 内 容 设置 为 红色 
echo "<span style='color:red'>"; 
} 
echo $i.".&nbsp;", 
echo unhtml(msubstr($info['title”],0,18));// 应 用 自 定义 函数 对 输出 的 内 容 进 
行 控制 
if(stren($info['title])>18X // 当 输出 内 容 的 长 度 超过 18 个 字符 时 用 “.…” 代 蔡 
Stho” 
} 
echo "(".str_replace("-","/",$info[createtime1]).")";// 将 输出 的 公告 时 间 中 的 
“-" 用 “/” 替代 
if($i==1X 
echo "</span>"; 
} 
?> 
</a> 
</div> 
<?php 
$i++; 
}while($info=mysqli_fetch_array($sql)); /do…while 循环 语句 结束 
} 
?> 
</div> 


在 首页 中 使 用 滚动 条 是 一 个 比较 不 错 的 方法 ， 可 以 增加 网 页 的 动态 效果 ， 增 加 网 页 的 观赏 性 ， 而 
且 不 会 影响 到 网 页 的 浏览 速度 。 


2.7.3 前 台 首 页 的 实现 过 程 


国 ”前 台 首 页 使 用 的 数据 表 : tb tell、 tb_soft、 tb bccd、tb bb、tb cjwt、tb bbs、tb sjxz 
开发 网 站 首页 主要 就 是 连接 数据 库 ， 从 数据 库 中 读 取 数 据 ， 最 后 应 用 循环 语句 将 数据 库 中 的 数据 
输出 到 前 台 页 面 。 由 于 使 用 的 代码 较 多 ， 而 且 多 数 都 是 重复 使 用 ， 所 以 这 里 只 给 出 首页 中 公告 发 布 模 


块 的 代码 。 
@ 
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公告 发 布 模块 主要 实现 从 数据 库 中 读 取 公 告 数据 ， 将 数据 在 首页 中 滚动 输出 ， 并 且 对 公告 信息 的 
长 度 进行 控制 ， 保 证 内 容 的 整齐 、 规 范 。 详 细 代码 可 以 参考 本 书 资源 包 中 的 TM\02\becty365\index.php 
文件 。index.php 文件 的 部 分 代码 如 下 : 


倒 程 04 代码 位 置 资源 包 \TM\02\bcty365\index.php 


<?php include_once("top.php"); /获取 头 部 文件 ?> 
…// 省 略 部 分 代码 
<?php /从 数据 库 中 读 取 公告 数据 
$sql=mysqli_query($conn,"select id,title,createtime from tb_tell order by createtime desc limit 0,10"); 
// 读 取 数 据 库 中 公告 表 中 的 数据 
$info=mysqli_fetch_array($sql); /执行 读 取 数 据 表 中 数据 的 语句 
if($info==falseX{ /| 判断 当 $info==false 时 执行 下 面 的 内 容 
?> 
<div align="center" style="height:25px;"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div> 
<?php 
Jelse{ // 如 果 返 回 值 不 为 空 ， 则 执行 下 面 的 do…while 循环 语句 
Si=1; // 定 义 变量 $i=1 
dof /| 执行 do…while 循环 语句 
?> 


<div class="scroll"> 
<a href="tellinfo.php?id=<?php echo Sinfo['id"];?>" class="a1"> 


<?php 
if($i==1X{ /| 判断 当 $i==1 时 ， 将 输出 的 内 容 设置 为 红色 
echo "<span style='color:red'>"; 
} 
echo $i.".&nbsp;"; 
0 echo unhtml(msubstr($info['title],0,18));// 应 用 自 定义 函数 对 输出 的 内 容 进 
行 控制 
@ if(strlen($info['title])>18)// 当 输出 内 容 的 长 度 超过 18 个 字符 时 用 “.…” 代 蔡 
echo" ...”; 
二 
@ echo "(".str_replace("-","/",$info[createtime"]).")";// 将 输出 的 公告 时 间 中 的 
“-” 用 “/” 赫 代 
if($i==1){ 
echo "</span>"; 
} 
> 
</a> 
</div> 
<?php 
$it+; /变量 自 加 1 
}while($info=mysqli_fetch_array($sql));//do…while 循环 语句 结束 
?> : 
i // 省 略 了 部 分 代码 
<?php 
include_once("bottom.php"); /1 调用 网 站 的 尾 文件 
?> 


@_ 
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< 代码 贴 二 
@ 应 用 unhtml0 和 msubstr0 自 定义 函数 去 除 输 出 字符 囊 中 的 空格 和 控制 输出 字符 囊 的 长 度 。 
@ strlen0: 获取 指定 字符 囊 的 长 度 。 
@ str_ replace0: 实现 字符 串 的 替换 。 


2.8 注册 模块 设计 





2.8.1 注册 模块 概述 


BCTY365 网 上 社区 系统 为 了 更 好 地 与 广大 网 民 朋 友 进 行 交流 和 沟通 ， 创 建 了 一 个 会 员 注册 模块 。 
通过 会 员 注册 模块 ， 可 以 有 效 地 对 用 户 信息 进行 
采集 ， 并 将 合法 的 用 户 信息 保存 到 指定 的 数据 表 TREE 
中 ， 实 现 与 用 户 的 长 期 沟通 和 交流 。 既 然 设置 了 


日 


会 员 注册 模块 ， 那 么 在 系统 中 就 要 为 会 员 提供 一 ma 
些 特殊 的 权限 。 在 本 系统 中 注册 会 员 可 以 拥有 如 a 
下 权限 : 在 本 社区 的 论坛 中 发 布 和 回复 帖子 、 在 en 
技术 支持 模块 中 发 表 留言 、 在 升级 下 载 模块 中 下 和 
载 软件 升级 包 等 ， 而 且 可 以 进行 修改 密码 和 找 回 


密码 。 会 员 注册 模块 的 运行 结果 如 图 2.23 所 示 。 ae 
2.8.2 ”注册 模块 技术 分 析 


在 会 员 注册 模块 中 ， 必 不 可 少 的 就 是 要 对 用 
户 输入 的 信息 进行 判断 ， 首 先 判断 用 户 填写 的 注册 信息 中 哪些 是 必须 填写 的 ， 哪 些 可 以 不 填写 ， 然 后 
进一步 判断 输入 的 信息 是 否 合理 合法 ， 例 如 ， 判 断 输入 邮编 的 格式 是 否 正 确 ， 判 断 输入 邮箱 的 格式 是 
否 正 确 等 。 对 表单 中 提交 的 数据 进行 判断 最 常用 的 办 法 就 是 使 用 JavaScript 脚本 , 也 可 以 使 用 正则 表达 
式 。 下 面 讲解 在 本 模块 中 如 何 通过 JavaScript 实现 表单 提交 数据 验证 。 

操作 原理 是 : 在 form 表单 中 调用 onsubmit 事件 ， 通 过 该 事件 调用 指定 的 JavaScript 脚本 ， 执 行 
chkinputO 自 定义 函数 ， 实 现 对 表单 中 提交 数据 的 验证 。 在 JavaScript 脚本 中 ， 实 现 对 表单 中 提交 的 数 
据 进行 判断 ， 判 断 输入 的 内 容 是 否 为 空 以 及 格式 是 否 正确 ， 如 果 正 确 则 继续 执行 ， 否 则 将 弹出 提示 对 








2.23 ”用户 注册 模块 的 运行 结果 





<script language="JavaScript" type="text/javascript"> 


function chkinput(formX{ /定义 一 个 函数 
ifform tel.value==""){ / 莽 断 tel 文本 框 中 的 值 是 否 为 空 
alert(" 请 填写 联系 电话 !); /如 果 为 空 ， 则 输出 “请 填写 联系 电话 !” 
form.tel.select(); // 返 回 到 tel 文本 框 


return(false); 
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if(form.email.value=="™"X{ // 判 断 email 文本 框 的 值 是 否 为 空 
alert(" 请 输入 E-mail 地 址 "); /如 果 为 空 ， 则 输出 “请 输入 E-mail 地 址 !” 
form.email.select(); /返回 到 email 文本 框 
return(false); 


站 
var i=form.email.value.indexOf("@"); 
var j=form.email.value.indexOf("."); 
1/ 进一步 判 断 邮 箱 的 格式 是 否 正确 ， 是 否 包含 “@” 和 “.” 
if((i<O)(j>0)G<0)X 
alert(" 请 输入 正确 的 E-mail 地 址 ""); 


form.email.select(); /| 返回 到 email 文本 框 
return(false); 
3 
return(true); /提交 表单 


} 
</script> 


上 述 代 码 中 ， 只 是 列举 了 JavaScript 中 的 部 分 内 容 ， 并 且 在 对 电话 号 码 进行 判断 时 ， 只 是 判断 其 是 
否 为 空 ， 没 有 进一步 判断 电话 号 码 的 格式 是 否 正确 。 如 果 想 要 更 加 准确 地 判断 电话 号 码 的 格式 是 否 正 
确 ， 可 以 采用 下 面 的 方法 : 通过 正则 表达 式 的 preg_match0 函 数 ， 在 表单 提交 处 理 页 中 对 电话 号 码 进 行 
判断 。 

preg_match0 函 数 的 语法 格式 如 下 : 





int preg_match(string pattern, string subject [, array matches [, int flags]]) 





preg_match0) 函 数 的 参数 说 明 如 表 2.7 所 示 。 
表 2.7 preg_match() 函 数 的 参数 说 明 


参数 说 了 明 
patterm | 必要 参数 。 需 要 匹配 的 正则 表达 式 
subject ”| 必要 参数 。 输 入 的 字符 串 


可 选 参数 。 输 出 的 搜索 结果 的 数组 ， 如 $out[0] 将 包含 与 整个 模式 匹配 的 结果 ，$out[]] 将 包含 与 第 一 个 捕 
获 的 括号 中 的 子 模式 所 匹配 的 结果 ， 依 次 类 推 


可 选 参数 。 标 记 PREG_ OFFSET CAPTURE 可 对 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 串 偏 移 量 ， 
本 标记 自 PHP 4.3.0 起 可 用 


matches 











flags 


通过 preg_match0 函 数 判 断 电话 号 码 的 格式 是 否 正确 的 方法 如 下 : 首先 定义 一 个 用 于 判断 电话 号 码 
格式 的 正则 表达 式 。 代 码 如 下 : 


/MQ\d{3})(d{8})sI(d{4})(d{7))sI"(d{4})(d{8})s/ 


正则 表达 式 的 功能 分 析 如 下 : 使 用 “^” 和 “$” 对 字符 串 进行 边界 的 限制 ， 对 区 号 从 字符 串 的 开 
始 进行 匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 ; 将 括号 “0” 中 的 内 容 作为 一 个 原子 使 用 ; 使 
用 “\d” 来 匹配 一 个 数字 ， 区 号 为 3 或 4 个 数字 ， 其 他 数字 为 7 或 8 个 ; 使 用 “{}” 来 对 前 字符 进行 重 
复 匹 配 ; 使 用 “|” 对 匹配 的 模式 进行 选择 ， 分 成 3 个 模式 。 


全 
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然后 将 该 正则 表达 式 应 用 到 preg_matchO) 函 数 中 , 对 表单 提交 的 电话 号 码 进行 判断 ， 如 果 正 确 则 继 
续 执 行 ， 否 则 弹出 提示 信息 ， 并 返回 到 表单 提交 页 。 代 码 如 下 : 


<?php 
Stel="0431-84978981"; // 定 义 一 个 电话 号 码 的 变量 
if(preg_match("/*(\d{3}-)(\d{8})$I^(\d{4}-)(d{7})SI^(d{4})(d{8})SI‘(d{11})$/", $tel, Scounts)X{ 
echo "您 输入 的 电话 号 码 格式 正确 中 ; // 输 出 字符 串 
Jelse{ 


echo "<script>alert( 您 输入 的 电话 号 码 的 格式 不 正确 山 );history back()</script>"; 
和 


?> 


2.8.3 注册 模块 的 实现 过 程 


国 注册 模块 使 用 的 数据 表 : tb User 

注册 模块 的 实现 过 程 非常 简单 ， 首 先 阅读 注册 服务 条 款 ， 然 后 填写 用 户 注册 的 用 户 名 和 密码 ， 提 
交 后 由 系统 判断 输入 的 用 户 名 是 否 被 占用 ， 如 果 未 被 占用 则 可 以 继续 注册 ， 填 写 详细 的 注册 信息 ， 将 
数据 提交 到 表单 处 理 页 进行 处 理 ， 最 后 将 用 户 注册 的 信息 保存 到 指定 的 数据 表 中 。 用 户 注册 模块 的 实 
现 过 程 主要 由 3 个 文件 完成 : registerphp 用 于 输出 注册 服务 条 款 ， 以 及 填写 注册 的 用 户 名 和 密码 ， 并 
且 判 断 注 册 的 用 户 名 和 密码 是 否 被 占用 ; getuserinfo .php 文件 用 于 填写 详细 的 注册 信息 ， 并 且 在 表单 中 
应 用 数字 验证 码 技术 ; savereginfo.php 文件 用 于 对 表单 中 提交 的 数据 进行 处 理 , 将 数据 保存 到 指定 的 数 
据 表 中 。 

在 savereginfo.php 文件 中 ， 首 先 连 接 数据 库 ， 然 后 获取 表单 中 提交 的 数据 ， 并 且 判 断 提 交 的 用 户 
名 是 否 被 占用 ， 最 后 将 提交 的 数据 进行 处 理 ， 并 将 数据 保存 到 指定 的 数据 表 中 。 程 序 代码 如 下 : 


倒 程 05 ”代码 位 置 ， 资源 包 \TM\02\becty365\savereginfo.php 
<?php 





header("Content-type: text/html; charset=utf-8"); // 设 置 文 件 编码 格式 

session_start(); /初始 化 session 变量 

include_once("conn/conn.php"); /| 连接 数据 库 

date_default_timezone_set("PRC"); // 设 置 时 区 

$usernc=trim($_POST[usernc]); // 获 取 注 册 的 用 户 名 

// 判 断 指定 的 用 户 名 是 否 存在 

$sql=mysqli_query($conn,"select usernc from tb_user where usernc=".$usernc.""); 

$info=mysqli_fetch_array($sql); / 按 指定 条 件 检索 数据 信息 

if($info!=falseX{ // 如 果 查 询 结果 不 为 空 ， 则 执行 以 下 操作 
echo "<script language='javascript'>alert( 对 不 起 ， 该 昵称 已 被 其 他 用 户 使 用 !");history.back();</script>"; 
exit; 

} 

$xym=trim($_POST[xym'); // 去 除 变 量 两 边 的 空格 

S$num=$_POST[num]: /接收 变量 值 


if(strval($xym)!=strval($num)X{ 
echo "<script>alert(' 验 证 码 输 入 错误 !");window.location.href='register.php';</script>"; 
EXit; 
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} 

// 对 表单 提交 的 数据 进行 处 理 

$truepwd=trim($_POST[pwd1]); // 获 取 真 实 密码 
S$pwd=md5($truepwd); 1/ 获取 加 密 密 码 
S$truename=trim($_POST[truename']); 1/ 获取 真实 姓名 
$email=trim($_POST[email]); /获取 邮箱 地 址 
$sex=$_POST[sex]; /获取 性 别 
Stel=trim($_POSTTtel]); /获取 电话 
$yb=trim($_POST[yb]); /获取 邮政 编码 
$qq=trim($_POST[qq]); /获取 QQ 
$address=trim($_POST[address']); // 获 取 地 址 
$question=trim($_POST[question]); /获取 提示 问题 
$answer=trim($_POST[answer]); // 获 取 问 题 答案 
S$ip=getenv("REMOTE_ADDR"); // 获 取 客 户 端 的 IP 
Slogintimes=1; /指定 访问 次 数 
$regtime=date("Y-m-j H:i:s"); // 获 取 时 间 
Slastlogintime=$regtime; 

$usertype=0; /指定 用 户 类 型 ， 默 认为 0 
$photo=$_POST["photo"]; // 获 取 头 像 

/将 表单 中 提交 的 数据 存储 到 数据 表 中 


if(mysqli_query($conn,"insert into tb_user(usernc,truename,pwd,email,sex,tel,qq,address,logintimes,regtime, 
lastlogintime,ip,usertype,yb,question,answer,truepwd,photo) 
values('$usernc','$truename','$pwd','$email','$sex','$tel','$qq','$address','$logintimes','$regtime','$lastlogintime', 
Sip','$usertype','$yb','$question','$answer','$truepwd','$photo")")X{ 

$_SESSION["unc"]=$usemc; 

echo "<script>alert( 注 册 成 功 ");window.location.href='index.php';</script>"; 


}else{ // 如 果 添 加 操作 失败 ， 则 给 出 提示 
echo "<script language='ijavascript'>alert(' 对 不 起 ,注册 失败 !");history.back();</script>"; 
exit;// 退 出 

} 

Pr 





2.9 技术 支持 模块 设计 





视频 讲解 


技术 支持 模块 主要 是 从 浏览 者 的 角度 进行 设计 ， 存 储 大 量 技术 问题 的 解决 方案 数据 ， 为 浏览 者 查 
阅 提供 方便 ， 而 且 设计 一 个 企业 与 客户 沟通 的 平台 ， 能 够 随时 了 解 客户 或 者 会 员 的 意见 和 需求 。 


2.9.1 技术 支持 模块 概述 


技术 支持 模块 主要 由 3 个 子 模块 组 成 ， 包 括 常见 问题 、 客 户 反馈 和 联系 方式 。 常 见 问题 模块 主要 
用 于 展示 编程 中 一 些 常 见 问题 的 解决 方案 或 者 方法 ， 为 浏览 者 解决 编程 中 的 疑难 问题 提供 方便 ; 客户 
反馈 模块 主要 用 于 收集 和 获取 来 自 客户 的 需求 和 意见 ;联系 方式 模块 主要 用 于 展示 企业 的 形象 和 具体 
的 联系 方式 。 


名 
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2.9.2 ”技术 支持 模块 技术 分 析 


技术 支持 模块 中 在 对 常见 问题 解决 方案 的 数据 进行 输出 时 ， 使 用 了 分 页 处 理 技 术 ， 该 技术 的 设计 思 
路 是 : 从 数据 库 中 读 取 数据 ， 获 取 数 据 总 量 ， 在 每 页 中 显示 20 条 数据 ， 根 据 数 据 总 量 和 每 页 显示 的 条 数 
对 数据 进行 分 页 处 理 ， 计 算出 有 多 少 页 和 当前 显示 的 页 码 ， 实 现 首页 、 上 一 页 、 下 一 页 和 尾 页 之 间 的 页 面 
跳 转 。 具 体 的 设计 思路 可 以 参考 cjwt.php 文件 中 的 代码 注释 和 代码 贴 士 。cjwt.php 文件 的 程序 代码 如 下 : 


倒 程 06 代码 位 置 ， 资源 包 \TM\02\becty365\cjwt.php 


<?php 
$sql=mysqli_query($conn,"select count(*) as total from tb_cjwt"); // 读 取 数 据 库 中 的 数据 
$info=mysqli_fetch_array($sql); // 返 回 数据 


@ S$total=$info[total]; 
// 判 断 字 段 total 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 


if($total==0X{ 
E 
<div align="center" style="height:38px; line-height:38px;"> 对 不 起 ， 暂 无 常见 问题 ! </div> 
<?php 
}else{ /如 果 不 为 空 ， 则 执行 下 面 的 内 容 
© if(!isset($_GET["page"]) || !is_numeric($_GETI["page"])X{ // 判 断 $_GET 获取 的 page 的 值 是 否 存在 
© $page=1; // 如 果 不 存在 ， 则 设置 变量 的 值 为 1 
}else{ 
9 Spage=intval($_GET["page"]); // 如 果 存 在 ， 则 获取 变量 $_GET 的 值 
} 
© $pagesize=20; /设置 变量 $pagesize, 每 页 显示 的 数据 量 为 20 
if($total% $pagesize==0){ // 如 果 变 量 的 值 为 0 
© $pagecount=intval($total/$pagesize); // 获 取 变 量 的 整数 值 
}else{ 
[7 Spagecount=ceil($total/$pagesize); // 如 果 不 为 0， 则 获取 实际 的 整数 值 


} 
// 读 取 数 据 库 中 的 数据 ， 按 照 时 间 进 行 降序 排列 
$sql=mysqli_query($conn,"select * from tb_cjwt order by createtime desc limit ".($page-1)*$pagesize. ", 
$pagesize "); 
‘while($info=mysqli_fetch_array($sql)X{ 
?> 
“» /| 省略 部 分 代码 
<div class="page-left"*>&nbsp;&nbsp; 共有 常见 问题 <?php echo Stotal;?> 条 &nbsp; 每 页 显示 <?php echo 
$pagesize;?> 条 &nbsp; 第 <?php echo $page;?> 页 / 共 <?php echo $pagecount;?> 页 </div> 
<div class="page-right"> 
© <ahref="<?php echo $_SERVER["PHP_SELF"]?>?page=1" class="a1"> 首 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 


if($page>1) /判断 如 果 页 码 大 于 1 
echo $page-1; /输出 前 一 页 
else 
echo 1; 


?>" class="a1"> 上 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 
if($page<$pagecount) // 如 果 页 码 小 于 总 页 数 


_ 园 
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echo $page+1; /输出 下 一 页 
else 
echo $pagecount; 
?>" class="a1"> 下 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php echo S$pagecount;?>” class="a1"> 尾 页 
</a>&nbsp;&nbsp;</div> 





Ah 代码 由 十 
@ S$total: 为 数据 库 中 数据 总 的 记录 数 。 
@ isset0 用 于 检测 变量 是 否 已 经 设置 is_numeric0 用 于 检测 变量 是 否 为 数字 或 者 数字 字符 囊 。 
@@ $page: 变量 为 页 码 中 的 第 几 页 。 
@intval0: 获取 变量 的 整数 值 。 
@@ $pagesize: 表示 在 每 页 中 显示 多 少 条 数据 。 
@ $pagecount: 表示 所 有 的 数据 可 以 分 成 多 少 页 。 
@ ceil0: 获取 变量 中 的 整数 值 ， 这 里 用 于 获取 页 码 的 整数 值 。 
@$_SERVER["PHP SELF"]: 服务 器 变量 ， 这 里 用 于 获取 网 页 的 链接 地 址 。 


2.9.3 ”常见 问题 的 实现 过 程 


国 常见 问题 使 用 的 数据 表 : tb_cjwt 

常见 问题 子 模块 实现 的 主要 功能 是 展示 出 数据 库 中 存储 的 有 关 编 程 中 遇 到 的 常见 问题 及 解决 方 
案 。 其 运行 结果 如 图 2.24 所 示 。 

该 模块 由 两 个 文件 组 成 ， 一 个 是 cjwtphp 文件 ， 
用 于 存储 创建 问题 数据 ， 详 细 内 容 可 以 参考 2.9.2 节 ; 
另 一 个 是 lookcjwt.php 文件 ， 用 于 输出 cjwt.php 文件 
中 对 应 问题 的 详细 介绍 和 解决 方案 。 代 码 如 下 : 


倒 程 07 代码 位 置 ， 资源 包 \TM\02\bcty365\lookcjwt.php 





图 2.24 常见 问题 模块 的 运行 结果 





<?php 
include_once("conn/conn.php"); /与 数据 库 建立 连接 
include_once("function.php"); // 调 用 自 定义 函数 


// 读 取 tb_cjwt 表 中 的 数据 
$sql=mysqli_query($conn,"select * from tb_cjwt where id=".$_GET["id"]."™™"); 
$info=mysqli_fetch_array($sql); 
?> 
<div class="lookcjwt-main"> 
<div class="question"><span><strong> 问 &nbsp;&nbsp; 题 : </strong></span> 
<div> 
<?php echo unhtml($info["question"]); // 输 出 问题 的 详细 内 容 ?> 
</div> 
</div> 
<div class="answer"><span><strong> 解 &nbsp;&nbsp; 答 : </strong></span> 
<div> 
<?php echo unhtml($info["answer"]); // 输 出 问题 的 解决 方案 ?> 
</div> 





© 
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</div> 
</div> 





2.9.4 客户 反馈 的 实现 过 程 


国 客户 反馈 使 用 的 数据 表 : tb_user、tb leaveword 
客户 反馈 子 模块 为 客户 提供 了 一 个 反馈 意见 和 提出 要 求 的 平台 ， 并 且 将 提交 的 信息 存储 到 数据 库 
中 。 其 运行 结果 如 图 2.25 所 示 。 





该 功能 只 对 本 网 站 中 的 会 员 开 通 ， 即 只 有 以 会 员 。 。 ， ”meeeaaamsnm 
身份 登录 的 用 户 才 具 有 反馈 信息 的 权限 ， 其 中 在 对 提 。。“” 
交 雪 间 的 编 节 处 理 目 合用 Jayasaiipt 脚本 来 县 证 过 单 i 


中 的 值 是 否 为 空 ， 而 且 还 使 用 了 数字 验证 码 技术 。 对 
表单 中 提交 的 数据 进行 处 理 是 在 saveleaveword.php 文 








件 中 完成 的 ， 该 文件 主要 用 于 对 表单 中 提交 的 数据 进 ee ee 
行 处 理 ， 将 数据 存储 到 数据 库 中 。 代 码 如 下 : 2.25 客户 反馈 模块 的 运行 结果 
倒 程 08 代码 位 置 资源 包 \TM\02\bcty365\saveleaveword.php 
<?php 
header("Content-type: text/html; charset=utf-8"); // 设 置 文 件 编码 格式 
session_start(); /初始 化 一 个 session 变量 
date_default_timezone_set("PRC"); 
$xym=$_POST[xym'; /| 获取 $_POST 提交 的 值 
if($xym!=$_SESSION["autonum"]X{ // 判 断 验证 码 是 否 正确 
echo "<script>alert(' 效 验 码 输入 错误 ! ');history.back();</script>"; 
exit; 
Stitle=$_POSTI["title"]; // 获 取 反 馈 信息 的 标题 
S$content=$_POST["content"]; // 获 取 反 馈 信息 的 内 容 
$type=$_POST["type"]; // 获 取 反 馈 信息 的 类 型 
include_once("conn/conn.php"); // 与 数据 库 建立 连接 
$sql=mysqli_query($conn,"select id from tb_user where usernc=".$_SESSION["unc"]."");// 读 取 数 据 库 中 数据 
$info=mysqli_fetch_array($sql); /获取 结果 集中 的 数组 
$userid=$info["id"]; 
// 向 数据 库 中 添加 数据 


if(mysqli_query($conn,"insert into tb_leaveword(userid,type,title,content,createtime) values('$userid','$type’, 
'$title','$content',".date("Y-m-j H:i:s").")")X{ 
echo "<script>alert(' 留 言 发 表 成 功 !');history.back();</script>"，// 添 加 操作 成 功 ， 给 出 提示 信息 
jelse{ 
echo "<script>alert( 留言 发 表 失败 ! ');history.back();</script>"，// 添 加 操作 失败 ， 给 出 提示 信息 
} 


?> 


2.9.5 单元 测试 
在 对 客户 反馈 模块 进行 测试 的 过 程 中 ， 当 提交 反馈 信息 时 ， 提 示 “ 留 言 发 表 失 败 !”。 经 过 初步 的 
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判断 , 问题 可 能 出 现在 向 数据 库 添加 数据 的 过 程 中 。 为 进一步 查找 出 问题 的 根源 , 在 saveleaveword.php 
页 中 应 用 了 mysqli_error0 函 数 ， 该 函数 可 获取 详细 的 错误 信息 ， 将 它 放置 在 insert 添加 语句 之 后 ， 然 
后 重新 运行 网 页 ， 运 行 结 果 如 图 2.26 所 示 。 














图 2.26 应 用 mysqli_error0 函 数 获取 的 错误 信息 
在 该 结果 图 中 可 以 看 出 ， 在 向 数据 表 中 添加 数据 时 字段 不 匹配 ， 经 过 仔细 检查 发 现 ， 原 来 是 在 编 












1 auerr(Sconn, "insert into tb_leaveword(userid, type title, content) values! 


echo “(script)oe-t( 留言 发 表 戌 功 ! ") :historyToeT ser 


et “4seriptyalsrt( 留言 
图 2.27 代码 中 的 错误 截图 
tt 语句 中 添加 createtime 字段 ， 重 新 运行 程序 ， 提 示 “ 留 言 发 表 成 功 !”。 





y. back() |%seript》” ;77 丘 加 线 作 闪 









2.10 在 线 订 购 模块 设计 


在 线 订购 模块 的 功能 是 实现 在 线 购买 企业 推出 的 软件 产品 ， 其 操作 的 流程 主要 通过 购物 车 和 订单 
管理 来 实现 。 


2.10.1 在 线 订购 模块 概述 
在 线 订购 的 功能 对 所 有 访问 网 站 的 人 开放 ， 没 有 任何 的 权限 限制 。 其 操作 流程 如 图 2.28 所 示 。 


直接 订购 


浏览 者 查看 信息 





图 2.28 在 线 订 购 模块 的 操作 流程 
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2.10.2 在线 订购 模块 技术 分 析 


在 线 订 购 管理 模块 中 ， 不 可 缺少 的 一 项 内 容 就 是 对 订单 进行 打印 。 下 面 就 来 讲解 一 下 订单 打印 功 
能 的 实现 方法 。 在 线 订购 管理 模块 中 运用 的 是 WebBrowser 打印 方法 。WebBrowser 是 正 内 置 的 浏览 器 
控件 ， 无 须 用 户 下 载 。 其 优点 是 客户 端 独立 完成 打印 目标 文档 的 生成 ， 减 轻 服 务 器 负荷 ， 缺点 是 源 文 
档 的 分 析 操 作 复 杂 ， 并 且 对 源 文档 中 要 打印 的 内 容 进行 约束 。 

下 面 介绍 WebBrowser 控件 的 具体 参数 ， 如 表 2.8 所 示 。 


表 2.8 WebBrowser 控件 的 具体 参数 说 明 





参数 名 称 说 ”了 明 
document.all. WebBrowser. Execwb(7.1): 表示 打印 预览 
document.all. WebBrowser.Execwb(6.1): 表示 打印 
document.all. WebBrowser. Execwb(6.6): 表示 直接 打印 
document.all. WebBrowser. Execwb(8.1): 表示 页 面 设置 
document.all. WebBrowser. Execwb(1.1): 打开 页 面 
document.all. WebBrowser Execwb(2.1): 关闭 所 有 打开 的 正 窗口 
document.all. WebBrowser. Execwb(4.1): 保存 网 页 
document.all. WebBrowser. Execwb(10.1): 查看 页 面 属 性 
document.all. WebBrowser. Execwb(17.1): 全 选 
document.all. WebBrowser Execwb(22.1): 刷新 
document.all. WebBrowser.Execwb(45,1): 关闭 窗 体 无 提示 


该 技术 的 实现 原理 是 : 首先 通过 onClick 事件 调用 一 个 JavaScript 脚本 ,然后 执行 openprintwindow() 
函数 ， 将 指定 的 变量 值 传递 到 订单 打印 页 中 〈printwindowphp)， 最 后 在 订单 打印 页 中 实现 打印 及 打印 
预览 的 功能 。 调 用 JavaScript 脚本 和 执行 openprintwindow0O 函 数 在 shopping_dd.php 页 中 完成 。 关 键 代 
码 如 下 : 


倒 程 09 代码 位 置 , 资源 包 \TM\02\becty365\shopping_dd.php 
<script language="javascript"> 

function openprintwindow(x,y,zX{ /定义 一 个 函数 ， 获 取 传递 的 参数 

// 通 过 window 对 象 中 的 open() 方 法 打开 一 个 新 窗口 ， 并 设置 其 属性 
@ window.open("printwindow.php?ddno="+x+"&pv="+z,"newframe","top=200,left=200,width=635, 
height="+(230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no, status=no"); 
} 

</script> 
<!-- 一 -一 -通过 onclick 事件 调用 JavaScript 脚本 ， 传 递 参 数值 。 当 参数 z 的 值 为 p 时 执行 打印 功能 一 一 一 - -> 
<td width="75"> <img src="images/bg_14_11.jpg" width="69" height="20" 
© onclick="javascript:openprintwindow('<?php echo base64_decode($_GET["ddno"]);?>','<?php echo 
S$gnum;?>",'p')" style="cursor:hand"/></td> 

一 > 


<[-- 一 一 通过 onclick 事件 调用 JavaScript 脚本 ， 传 递 参数 值 。 当 参数 z 的 值 为 v 时 执行 打印 预览 功能 一 


eo) 
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<td width="90"><img src="images/bg_14_12.jpg" width="69" height="20" onclick="javascript:openprintwindow 
(<?php echo base64_decode($_GET["ddno"]);?>','<?php echo $gnum;?>",'v'")" style="cursor:hand"/></td> 


人 代码 贴 士 
@ open0: 打开 一 个 窗口 ， 可 以 设置 其 URL、 名 称 、 大 小 、 按 钮 以 及 其 他 属性 。 该 方法 的 基本 语法 结构 如 下 : 


window.open(url,name,features,replace) 





其 中 ，url 是 要 在 新 窗口 中 打开 文档 的 URL 地 址 ; name 是 要 打开 窗口 的 名 字 ， 用 HTML 链接 的 target 属性 进行 定 
位 时 会 有 用 ; features 是 一 个 用 过 号 分 隔 的 字符 串 ， 列 举 窗口 的 特征 ; Teplace 是 一 个 可 选 的 Boolean 值 ， 指 出 是 否 允 许 
URL 替换 窗口 的 内 容 。 

@ onclick0: 表明 某 元 素 被 鼠标 单 击 触 发 的 事件 。 


订单 的 打印 和 打印 预览 的 功能 在 printwindow.php 页 中 完成 ， 首 先 编写 一 个 实现 打印 预览 功能 的 
JavaScript 脚本 ， 然 后 建立 HTML 的 object 标签 ， 调 用 WebBrowser 控件 ， 最 后 获取 变量 传递 的 值 ， 当 
变量 的 值 为 p 时 执行 打印 功能 ， 当 变量 的 值 为 v 时 执行 打印 预览 的 功能 。printwindow.php 页 的 关键 代 
码 如 下 : 


倒 程 10 ”代码 位 置 : 资源 包 \TM\02\bcty365\printwindow.php 





<script> 
function printview(){ /定义 一 个 函数 
@ document.all.WebBrowser1.ExecWB(7,1); /执行 WebBrowser 控件 ， 实 现 打 印 预 览 
window.close(); // 关 闭 窗口 
3 
</script> 
<I-- -一 - 建立 HTML 的 object 标签 ， 调 用 WebBrowser 控件 -一 一 一 一 一 -一 一 -> 





<object ID='WebBrowser1 WIDTH=0 HEIGHT=0 
CLASSID='CLSID:8856F961-340A-11D0-A96B-00C04FD705A2'></object> 
< 一 一 一 一 一 一 一 一 -一 在 body 中 调用 onload 事件 ， 执 行 打印 操作 -一 一 一 一 一 一 一 一 一 一 
四 <bodytopmargin="0" leftmargin="0" bottommargin="0" onLoad=" 
<?php 

if($_GET["pv"]=="p"}{ // 判 断 当 变 量 值 为 p 时 ， 执 行 打印 操作 
?> 
© window.print(); 
<?php 

}elseif($_GET["pv"]=="V"X/ // 判 断 当 变量 值 为 v 时， 执行 打印 预览 操作 
?> 

printview() 
<?php} ?> 
> 


Ah 代码 贴 十 
@ document.all.WebBrowser1.ExecWB(7.1): WebBrowser 控件 中 的 参数 ， 表 示 打 印 预览 。 
@ onLoad: 表明 某 对 象 已 载 入 窗口 。 
@ print0: window 对 象 中 的 一 个 方法 ， 实 现 打印 的 功能 。 


订单 打印 操作 的 运行 结果 如 图 2.29 所 示 。 


@ 


第 2 章 BCTY365 网 上 社区 (Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


1365 社 区 网 ,二 


新 里 











图 229 订单 打印 操作 的 运行 结果 
2.10.3 ”购物 车 的 实现 过 程 


回 ”购物 车 使 用 的 数据 表 : tb_bccd 
购物 车 的 功能 是 临时 储存 用 户 选 购 的 商品 ， 用 户 可 以 对 购物 车 中 的 商品 进行 添加 、 修 改 、 删 除 和 
更 新 操作 ， 也 可 以 选择 进行 结算 。 其 运行 的 结果 如 图 2.30 所 示 。 








图 2.30 购物 车 的 运行 结果 图 


购物 车 功能 实现 的 第 一 步 是 为 想 要 购买 产品 的 用 户 分 配 一 辆 购物 车 ， 使 其 能 够 记录 自己 已 经 选 购 
的 产品 。 其 工作 的 原理 与 超市 中 顾客 使 用 购物 车 进行 购物 是 相同 的 ， 只 是 这 里 使 用 的 不 是 真正 意义 上 
的 购物 车 ， 而 是 两 个 session 变量 ， 一 个 存储 用 户 选 购 商品 的 ID 〈$goodsid)， 另 一 个 存储 用 户 选 购 该 
商品 的 数量 〈$goodsnum)。 如 果 用 户 在 一 次 购物 中 选 购 多 种 不 同类 的 商品 ， 则 使 用 @ 对 不 同类 商品 的 
不 同 DD 和 数量 进行 分 割 。 例 如 ， 用 户 选 购 的 不 同类 商品 id 为 1、2、3， 则 session 变量 $goodsid 中 存 
储 的 值 为 “1@2@3@” 其 中 同一 种 商品 不 能 购买 两 次 ， 如 果 想 要 购买 多 个 同 种 产品 ， 可 以 在 购物 车 中 
更 改 购买 商品 的 数量 。 在 本 案例 中 ， 购 物 车 的 分 配 功能 通过 shopping_cart_first.php 文件 来 完成 ， 该 文 
件 的 代码 如 下 : 


倒 程 11 代码 位 置 : 资源 包 \TM\02\bcty365\shopping_cart first.php 


<?php 
header("Content-type: text/html; charset=utf-8"); 1/ 设置 文件 编码 格式 
@ session_start(); /初始 化 session 变量 


if(!isset($_SESSION["goodsid"]) && lisset($_SESSION["goodsnum"])X ”// 判 断 session 变量 中 的 值 是 否 为 空 


四 
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$_SESSION["goodsid"]=$_GETT["id"]."@"; /如 果 为 空 ， 则 将 商品 的 ID 赋 给 变量 
$_SESSION["goodsnum"]="1@"; // 将 商品 数量 设置 为 “1@” 
jelsef 


© $array=explode("@",$_SESSION["goodsid"]); /如 果 不 为 空 ， 则 使 用 @ 分 割 不 同 的 商品 ID 
// 判 断 如 果 获取 的 ID 在 session 变量 中 已 经 存在 ， 则 提示 该 商品 已 经 被 放 入 购物 车 
© if(in_array($_GET["id"], $array)}{ 
echo "<script>alert(' 该 编程 词典 已 经 被 放 入 购物 车 ! ');history.back();</script>"; 


exit; 
$_SESSION["goodsid"].=$_GETT["id"]."@"; /为 session 变量 赋值 
$_SESSION["goodsnum"].="1@"; /为 session 变量 赋值 


} 

// 将 商品 放 入 购物 车 中 ， 并 跳 转 到 购物 车 页 

echo "<script>window.location.href='shopping_cart.php';</script>"; 
?> 





< 代码 贴 十 

@ session_ start0: 初始 化 session 变量。 

@ explode0: 将 字符 囊 依 指定 的 字符 串 或 字符 进行 分 割 。 返 回 由 字符 囊 组 成 的 数组 ， 每 个 元 素 都 是 string 的 一 个 子 
串 ， 它 们 被 字符 串 separator 作为 边界 点 分 隔 出 来 。 如 果 设置 了 limit 参数 ， 则 返回 的 数组 包含 最 多 limit 个 元 素 ， 而 最 后 
那个 元 素 将 包含 string 的 剩余 部 分 ;如果 separator 为 空 字符 囊 (" )，explode0 函 数 将 返回 false; 如 果 separator 所 包含 
的 值 在 string 中 找 不 到 ， 那么 explode0O 函 数 将 返回 包含 string 单个 元 素 的 数组 ; 如 果 参 数 limit 是 负数 ， 则 返回 除了 最 后 
的 -limit 个 元 素 外 的 所 有 元 素 。 

四 in_array0: 在 指定 的 数组 中 搜索 某 个 值 ， 如 果 找到 则 返回 tue， 否 则 返回 false。 


在 实现 购物 车 的 分 配 和 添加 商品 的 功能 后 ， 接 下 来 要 做 的 就 是 查看 购物 车 中 的 商品 ， 即 实现 购物 
车 中 商品 展示 的 功能 。 在 购物 车 的 商品 展示 中 ， 可 以 实现 很 多 的 操作 ， 如 清空 购物 车 、 删 除 购买 商品 、 
更 改 购买 商品 数量 、 继 续 购 物 和 结算 。 

购物 车 商品 展示 的 功能 主要 通过 shopping_cart.php 文件 来 完成 ， 首 先 从 session 变量 中 读 取 商品 的 
ID 和 数量 , 然后 根据 商品 的 ID 循环 输出 购物 车 中 的 商品 , 最 后 以 商品 ID 为 标识 符 设置 不 同 的 超 链 接 ， 
执行 删除 商品 或 者 更 改 购买 商品 数量 等 操作 。 代 码 如 下 : 


倒 竹 12 代码 位 置 ， 资源 包 \TM\02\becty365\shopping_cart.php 


<table class="cart-table"> 
<tr> 
<td width="330" height="22" bgcolor="#CCCCCC"><div align="center"> 商 品名 称 </div></td> 
<td width="155" bgcolor="#CCCCCC"><div align="center"> 单 价 (元) </div></td> 
<td width="87" bgcolor="#CCCCCC"><div align="center"> 数 量 (个 ) </div></td> 
<td width="143" bgcolor="#CCCCCC"><div align="center"> 操 作 </div></td> 
</tr> 
<?php 
$array=explode("@",isset($_SESSION["goodsid"])?$_SESSION["goodsid"]:");// 读 取 session 变量 中 
的 商品 ID， 以 @ 进 行 分 割 
S$arraynum=explode("@",isset($_SESSION["goodsnum"])?$_SESSION["goodsnum"]:");// 读 取 session 
变量 中 的 商品 数量 ， 以 @ 进 行 分 割 


$markid=0; // 创 建 变量 ， 初 始 值 为 0 
局 
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for($i=0;$i<count($array);$i++X{ /应 用 for 循环 语句 循环 输出 商品 ID 的 值 
if($array[$i]!="™"X{ // 判 断 如 果 商 品 ID 的 值 不 为 空 
$markid++; // 增 加 变量 $markid 的 值 
} 
} 
if($markid==0X{ // 判 断 如 果 变 量 $markid 的 值 为 空 ， 则 输出 下 面 的 内 容 


?> 


<tr> 
<td height="22" colspan="4" bgcolor="#FFFFFF"><div align="center"> 对 不 起 ， 您 的 购物 车 中 暂 无 商品 信 
息 ! </div></td> 


</tr> 
<?php 
}else{// 如 果 $markid 的 值 不 为 空 ， 则 执行 下 面 的 内 容 
S$totalprice=0; // 创 建 变量 $totalprice， 初 始 值 为 0 
for($i=0;$i<count($array);$i++)( /循环 输出 数组 中 的 商品 ID 值 
if$array[$i]!=""){ 
/根据 获取 商品 ID 的 值 ， 从 数据 库 中 获取 对 应 产品 的 信息 
$sqlcart=mysqli_query($conn,"select * from tb_bccd where id=".$array[$i].""); 
Sinfocart=mysqli_fetch_array($sqlcart) 
?> 
<tr> 


<form name="form<?php echo $array[$i]?>" method="post" action="changegoodsnum.php"> 
<td height="22" bgcolor="#FFFFFF">&nbsp;<?php echo unhtml($infocart["bccdname"]);?></td> 
<td height="22" bgcolor="#FFFFFF"><div align="center"><?php echo number_format($infocart 
["price"],2);?></div></td> 
<td height="22" bgcolor="#FFFFFF"><div align="center"><input type="text" name="goodsnum" value= 
"<?php echo $arraynum["$i"];?>" class="inputcss" size="8" ><input type="hidden" name="id" value="<?php 
echo $infocart["id"];?>" ></div></td> 
<td height="22" bgcolor="#FFFFFF"><div align="center"><a href="javascript:form<?php echo $array[$i]?>. 
submit();”class="a1"> 更 改 数量 </a>&nbsp;|&nbsp;<a href="delgoods.php?id=<?php echo S$infocart["id"];?>" 
class="a1"> 删 除 该 项 </a></div></td> 
</form> 
</tr> 
<?php 
Stotalprice+=$infocart["price"]*$arraynum["$i"; 


} 
?> 
</table> 





2.10.4 商品 订单 的 实现 过 程 


国 商品 订单 使 用 的 数据 表 : tb bccd、tb_ dd、tb city 

在 确定 所 要 购买 的 商品 之 后 ， 接 下 来 要 做 的 就 是 进行 购物 结算 ， 填 写 用 户 购物 订单 ， 将 订单 保存 
到 数据 库 中 ， 并 且 随 机 生成 一 个 订单 号 ， 作 为 订单 的 唯一 标识 。 生 成 订单 的 运行 结果 如 图 2.31 所 示 。 

订单 提交 以 后 用 户 可 以 选择 汇款 的 方式 : 一 是 选择 网 上 支付 ， 那 么 将 跳 转 到 企业 指定 的 网 上 银行 
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进行 汇款 ， 汇 款 的 操作 将 在 企业 指定 的 网 上 银行 中 进行 ， 这 里 不 做 讲解 ， 二 是 选择 到 指定 的 银行 向 企 
业 提供 的 账号 中 汇款 。 企 业 将 在 收 到 汇款 后 按照 用 户 指定 的 地 址 和 方式 将 产品 送 到 。 商 品 订单 的 生成 
和 处 理由 shopping_cart_getuserinfo php 和 savebuyuserphp 文件 来 完成 。 














地 直 ， 卡 本 市 二 3 
Ra 
mh 





图 2.31 生成 订单 的 运行 结果 


订单 处 理由 savebuyuserphp 完成 ， 首 先 连接 数据 库 ， 随 机 生成 一 个 订单 号 ， 然 后 获取 购物 车 中 的 
商品 信息 ， 最 后 将 商品 信息 和 订单 号 存储 到 数据 库 中 。 代 码 如 下 : 


倒 程 13 代码 位 置 ， 资源 包 \TM\02\bcty365\savebuyuser.php 





<?php 

header("Content-type: text/html; charset=utf-8"); // 设 置 文件 编码 格式 
session_start(); /| 初始化 session 变量 
include_once("conn/conn.php"); /| 连接 数据 库 
date_default_timezone_set("PRC"); 
$ddnumber=substr(date("YmdHis"),2,8).mt_rand(100000,999999); // 随 机 生成 订单 号 


$sql=mysqli_query($conn,"select * from tb_city where id=".$_POST["city"].""); // 读 取 数 据 库 中 的 城市 信息 
$info=mysqli_fetch_array($sql); 
if($_POST[shfs]=="1"X{ // 判 断 用 户 选择 的 送 货 方式 
$yprice=$info[ pt]; 
$shfs=" 普 通 邮递 "; 
Jelseif($_POST['shfs]=="2"}{ 
$yprice=$info[kd]; 
$shfs=" 邮 政 特快 专递 EMS"; 


} 
S$array=explode("@",$_SESSION["goodsid"]); /以 @ 来 分 割 session 变量 中 存储 的 商品 ID 
S$arraynum=explode("@",$_SESSION["goodsnum"]); /以 @ 来 分 割 session 变量 中 存储 的 商品 数量 
S$totalprice=0; 
for($i=0;$i<count($array);$i++ 交 // 循 环 读 取 数 组 中 商品 的 ID 
记 $array[$i]!="”){ 
$sqlcart=mysqli_query($conn,"select * from tb_bccd where id=".$array[$i].”); 
Sinfocart=mysqli_fetch_array($sqlcart); 
S$totalprice+=$infocart["price"]*$arraynum["$i"]; 
} 
} 
Stotalprice=$totalprice+ $yprice; 1/ 获取 汇款 金额 


全 
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/将 表单 中 提交 的 数据 存储 到 数据 库 中 

if(mysqli_query($conn,"insert into tb_dd(ddnumberrecusersex,address,yb,qq,emailmtel,gtel,shfs,spc,slc,yprice， 
totalprice,createtime,cityid) 
values(".$ddnumber.",".$_POST["recuser"].",".$_POST["sex"].",".$_POST["address"].",".$_POST["yb"].",".$ 
_POST["qqq].",".$_POST['email].",".$_POST["mtel]",".$_POST["gtel]",".$shfs.",".$_SESSION["goodsidq] 
",".$_SESSION["goodsnum"].",".$yprice.",".$totalprice.",”".date("Y-m-d Hii:s").",".$_POSTT'city"].")")X{ 
unset($_SESSION["goodsid"]); // 注 销 session 变量 goodsid 
unset($_SESSION["goodsnum"]); /注销 session 变量 goodsnum 

echo "<script>window.location.href='shopping_dd.php?ddno=".base64_encode($ddnumber).";</script>"; 

jelsef 

echo "<script>alert(' 订 单 信 息 保存 失败 ， 请 重 试 ! ');</script>"; 

} 


?> 


2.10.5 ”单元 测试 


在 在 线 订 购 模块 的 测试 过 程 中 ， 当 执行 删除 购物 车 中 某 个 商品 的 操作 时 ， 发 现 该 项 操作 没 能 正确 
执行 ， 指 定 的 商品 没有 被 删除 。 首 先 ， 从 该 功能 实现 的 思路 入 手 分 析出 错 原因 ， 并 且 仔细 核对 其 中 使 
用 的 变量 以 及 函数 是 否 正确 。 

删除 商品 的 操作 是 根据 商品 的 ID 来 进行 处 理 的 ， 当 单 击 “ 删 除 ” 超 链接 时 ， 将 指定 商品 的 了 D 通 
过 变量 $ides 传 入 到 执行 删除 商品 操作 的 文件 delgoods.php 中 ， 然 后 在 该 文件 中 获取 变量 $ides 传 入 的 
ID 值 ， 并 且 根 据 ID 值 完成 删除 购物 车 中 指定 商品 的 操作 。 在 查看 该 功能 实现 的 过 程 中 发 现 ， 定 义 的 
变量 $ides 与 delgoods.php 文件 中 $_GET["id"] 获 取 的 变量 值 不 一 致 ， 从 而 导致 删除 商品 的 操作 没有 正确 
执行 ， 代 码 的 出 错位 置 如 图 2.32 所 示 。 
shopping_cart.php 文件 中 执行 删除 操作 的 内 容 


ight="22” bgcolor="WFFFFFF” 
div align"center” 
‘a href-=" javascript:fgrac2ahp cd6jgR8gS -Rh :bcit0) :” class="a1") 更 改 亲生 c/a 
ph ect tertererred"] :9>”class="al”) 旨 队 该 项 (/a) 
















定义 传递 这 值 的 变量 
定义 与 获取 的 变量 值 不 一 致 | 文件 中 执行 删除 操作 的 内 容 




















er, SSESSIONTgoodsid-]) 


获取 变量 传递 的 值 | ” 














图 2.32 代码 中 的 错误 显示 
查找 出 错误 原因 ， 将 定义 的 变量 与 获取 的 变量 值 统一 使 用 id， 保 存 文件 ， 重 新 运行 程序 ， 删 除 操 


作 可 以 正常 运行 。 
OD 


PHP 项 目 开发 全 程 实录 (第 4 版 ) 








2.11 社区 论坛 模块 设计 











| 视频 讲解 




















社区 论坛 模块 为 网 站 的 浏览 者 提供 一 个 交流 的 平台 ， 以 此 来 扩大 网 站 的 影响 力 ,汇聚 更 多 的 人 气 ， 
宣传 企业 形象 ， 推 广 企业 产品 。 二 J 
2.11.1 社区 论坛 模块 概述 管理 页 而 

















社区 论坛 模块 为 浏览 者 、 会 员 、 客 户 和 企业 之 
间 提 供 一 个 大 的 交流 平台 ， 根 据 身份 的 不 同 ， 分 别 
给 予 不 同 的 操作 权限 。 社 区 论坛 模块 的 操作 流程 如 
图 2.33 所 示 。 

在 本 论坛 中 ， 浏 览 者 只 能 够 查看 帖子 ;注册 会 
员 既 可 以 查看 帖子 ， 也 可 以 发 布 和 回复 帖子 ， 管 理 
员 则 具有 发 布 、 回 复 、 查 看 和 删除 的 权限 。 


2.11.2 ”社区 论坛 模块 技术 分 析 图 2.33 社区 论坛 流程 图 


在 社区 论坛 模块 的 实现 过 程 中 , 通过 JavaScript 脚本 和 下 拉 列 表 框 的 结合 实现 一 个 不 同 版 块 之 间 快 
速 跳 转 的 功能 ， 从 而 能 够 更 加 灵活 、 方 便 地 实现 不 同 版 块 之 间 的 跳 转 。 

下 面 分 析 该 技术 是 如 何 实现 的 。 该 技术 的 实现 综合 3 个 方面 的 内 容 ， 以 一 个 下 拉 列 表 框 为 主 ， 通 
过 PHP 语句 从 数据 库 中 读 取 数 据 作为 下 拉 列 表 框 的 值 ， 应 用 onchange 事件 来 调用 JavaScript 脚本 ， 实 
现 不 同 版 块 之 间 的 跳 转 。 这 里 以 bbs_top.php 文件 中 的 快速 跳 转 功能 为 例 进行 分 析 。 关 键 代 码 如 下 : 


倒 程 14 ”代码 位 置 ， 资源 包 \TM\02\becty365\bbs_top.php 


<!- 创 建 一 个 下 拉 列 表 框 ， 指 定名 称 为 select_type， 并 且 设 置 其 属性 ， 通 过 onchange 事件 来 调用 JavaScript 脚 
本 文件 ， 实 现 页 面 跳 转 -> 
<select name="select_type" class="inputcss" 
@ onChange="javascript:window.location=this.options[this.selectedlndex].value;" > 
<?php 
// 通 过 PHP 语句 从 数据 库 中 读 取 数 据 ， 使 用 数据 的 ID 作为 下 拉 列 表 框 的 值 ， 使 用 数据 的 标题 title 作为 下 拉 
列表 框 显示 的 内 容 
$sql=mysqli_query($conn,"select * from tb_type_small order by createtime desc"); 
Sinfo=mysqli_fetch_array($sql); 
if($info=="™"}{ 
echo "<option> 暂 无 讨论 区 </option>"; 
Jelse{ 
echo "<option>- 版 块 快速 跳 转 -</option>"; 
dof /应 用 do…while 循环 语句 输出 下 拉 列 表 框 中 的 值 
2 echo "<option value="bbs_list.php?id=".$info["id].">".$info['title]."</option>"; 
}while($info=mysqli_fetch_array($sql)); /应 用 do…while 循环 语句 结束 
局 
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?> 
</select> 





Ah 代码 由 二 
@ onChange: 某 元 素 失去 焦点 ， 并 且 从 用 户 最 后 一 次 访问 以 来 ， 其 值 已 经 改变 。 
location: 用 于 访问 窗口 的 当前 定位 (URL )， 既 可 被 读 取 ， 又 可 被 置换 ， 可 以 通过 其 实现 某 个 页 面 的 定位 或 者 更 新 。 
@ <option value="… '>…</option>: 下 拉 列 表 框 中 输出 的 值 ， 以 及 显示 的 内 容 。 


该 技术 实现 的 运行 结果 如 图 2.34 所 示 ， 它 将 实现 从 JSP 版 块 跳 转 到 PHP 版 块 。 


前 信 天 )》 社 花坛 ET | 
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图 2.34 ”版块 跳 转 功能 的 运行 结果 
2.11.3 ”论坛 分 类 的 实现 过 程 


国 ” ”论坛 分 类 使 用 的 数据 表 : tb_bbs、tb_type_big、tb_type_small 


论坛 分 类 可 以 分 为 两 类 : 一 是 论坛 中 大 
的 版 块 分 区 ， 分 为 综合 信息 讨论 区 、 操 作 系 
统 、 程 序 设计 交流 区 和 数据 库 技术 4 个 版 块 ， 
其 数据 存储 于 tb_type_big 数据 表 中 。 二 是 对 
应 不 同 的 版 块 中 不 同 语言 和 技术 的 分 类 ， 分 se 
为 6 种 ， 其 数据 存储 于 tb_type_small 表 中 。 > 
论坛 分 类 的 运行 结果 如 图 2.35 所 示 。 | 。 
































论坛 分 类 的 实现 原理 很 简单 ， 首 先 从 | pr 
tb_type_big 表 中 读 取 6 个 版 块 中 的 数据 ,进行 sae somite wo me : 
循环 输出 ， 然 后 在 版 块 中 获 套 循环 ， 用 于 输出 | 
不 用 语言 的 分 类 数据 。 该 功能 主要 通过 sr a A 
bbs_index.php 文件 来 完成 ,bbs_index.php 文件 Sic a 

a 坛 

的 程序 代码 如 下 ; 图 2.35 论坛 分 类 的 运行 结果 

倒 程 15 代码 位 置 资源 包 \TM\02\bcty365\bbs_index.php 

<?php 

include_once("top.php"); // 调 用 网 站 头 文件 

include_once("bbs_top.php"); // 调 用 社区 论坛 的 头 文件 

?> 

<div class="main-box"> 

<?php 
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// 循 环 输出 数据 表 tb_type_big 中 的 4 个 版 块 数据 
$sql=mysqli_query($conn,"select * from tb_type_big order by createtime desc"); 
$info=mysqli_fetch_array($sql); 
if$info==false){ /如果 返回 值 为 false， 则 执行 下 面 的 内 容 
cg 
pe 省 略 了 部 分 代码 
<?php 
Jelse{ // 如 果 返 回 值 为 true， 则 执行 do…while 循环 语句 
A* 外 部 嵌 套 循环 ， 输 出 论坛 中 的 版 块 分 类 数据 */ 
Ly do{ 


ey 省 略 了 部 分 代码 
<table cellspacing="1" bgcolor="#6EBEC7"> 
<?php 
/循环 输出 tb_type_small 表 中 的 不 同 语言 和 技术 的 分 类 数据 
$sql1=mysqli_query($conn,"select * from tb_type_small where bigtypeid=". $info["id"]."™"); 
Sinfo1=mysqli_fetch_array($sql1); 
if($info1==falseX{ /| 判断 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 


> 
ne 省 略 了 部 分 代码 
<?php 
jelse{ /如果 返 回 值 为 true， 则 执行 下 面 的 内 容 ， 输 出 该 版 块 中 对 应 语言 和 技术 的 帖子 详细 信息 


es 省 略 了 部 分 代码 
<?php /内 部 嵌 套 循环 ， 输 出 不 同 语言 和 技术 的 分 类 %/ 
@ dof ?> 


We 省 略 了 部 分 代码 
<?php 
}while($info1=mysqli_fetch_array($sql1)); 
A* 内 部 幅 套 循环 结束 */ 
} 


?> 
</table> 
</div> 
<?php 
}while($info=mysqli_fetch_array($sql)); 
/外 部 嵌 套 循环 结束 ， 对 版 块 中 的 大 类 进行 输出 
} 


?> 
<?php include_once("bottom.php"); ?> 


< 人 代码 贴 十 
@ do…while 循环 语句 ， 对 论坛 中 大 的 版 块 分 类 进行 循环 输出 。 
@ do…while 循环 语句 ， 对 论坛 中 一 个 版 块 的 不 同 语言 和 技术 进行 循环 输出 。 


和 注意 在 应 用 do…while 循环 语句 时 ，while 后 的 分 号 不 能 省 略 。 


局 
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2.11.4 ”论坛 帖子 浏览 的 实现 过 程 


国 ”论坛 帖子 浏览 使 用 的 数据 表 : tb bbs、tb_user、tb_reply 

论坛 帖子 浏览 主要 输出 指定 帖子 的 详细 信息 ， 包 括 发 帖 人 、 用 户 级 别 和 注册 的 时 间 ， 以 及 帖子 的 
主题 、 内 容 和 发 帖 时 间 ， 包 括 上 传 的 图 片 。 本 模块 是 用 户 权限 使 用 体现 的 最 明显 地 方 ， 可 以 分 为 3 种 
情况 ， 第 一 以 浏览 者 进行 登录 ， 只 能 是 浏览 帖子 的 内 容 ， 没 有 其 他 权限 ;第 二 以 会 员 进行 登录 ， 可 以 
对 帖子 进行 回复 ， 发 表 自 己 的 看 法 ， 第 三 以 管理 的 身份 进行 登录 ， 不 但 可 以 回复 帖子 ， 而 且 可 以 对 任 
何人 发 布 和 回复 的 帖子 进行 删除 和 顶 帖 的 操作 。 下 面 就 来 看 一 下 以 管理 员 身 份 进行 登录 时 都 具备 哪些 
权限 ， 运 行 结果 如 图 2.36 所 示 。 
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图 2.36 管理 员 浏 览 帖子 的 结果 图 


论坛 帖子 浏览 的 功能 通过 bbs_ lookbbsphp 文件 完成 , 首先 根据 传递 的 ID 值 读 取 指定 的 帖子 数据 , 然 
后 判断 登录 用 户 的 类 型 ， 最 后 根据 用 户 不 同 的 类 型 执行 不 同 的 操作 。 代 码 如 下 : 


倒 程 16 代码 位 置 ， 资源 包 \TM\02\bcty365\bbs_lookbbs.php 

<?php 

include_once("top.php"); 

include_once("bbs_top.php"); 

// 根 据 $_GET 传递 的 数据 获取 tb_bbs 中 的 数据 
$sqlb=mysqli_query($conn,"select * from tb_bbs where id=".$_GET["id"]."™™"); 
$infob=mysqli_fetch_array($sqlb); 

/根据 $_GET 传递 的 数据 获取 tb_user 中 的 数据 
$sql4=mysqli_query($conn,"select * from tb_user where id=".$infob["userid].”); 
$info4=mysqli_fetch_array($sql4); 





?> 
人 /省 略 了 部 分 HTML 代码 
<li> 用 户 级 别 : 
<?php 
/根据 用 户 信息 表 tb_user 中 字段 usertype 的 值 判断 该 用 户 的 类 型 
// 如 果 值 为 1 则 是 管理 员 ， 值 为 2 则 是 后 台 管理 员 ， 值 为 0 则 是 普通 会 员 
© if($info4["usertype"]=="1") echo "管理 员 ";,else echo "普通 会 员 "; 
?> 
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</li> 

1/ 省略 了 部 分 HTML 代码 

<div><img src="images/lt_15_11.jpg" width="25" height="25"> 
<span><?php echo $infob["createtime"];?></span></div> 


<div> 
<?php 
// 判 断 tb_bbs 表 中 的 字段 photo 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 
© if($infob['photo"]!=""){ 
$photos=substr($infob["photo],2,70); /获取 图 片 在 服务 器 中 的 存储 路 径 
echo (stripslashes($infob["content"])); /| 输出 帖子 的 内 容 
echo "<img src=\"$photos\">"; /根据 获取 的 图 片 路 径 ， 输 出 服务 器 中 的 图 片 
Jelse{ // 如 果 tb_bbs 表 中 的 图 片 字段 photo 为 空 ， 则 执行 下 面 的 内 容 
目 echo (stripslashes($infob["content"])); /只 输出 帖子 的 内 容 
} 
?> 
</div> 
<div> 


<img src="images/lt_15_9.jpg" width="72" height="23" id="button_show_bbs" style="cursor: 
hand" onClick="<?php 
if(!isset($_SESSION["unc"])X{ 
echo "javascript:alert(' 请 先 登录 本 站 ， 然 后 回复 帖子 ! ');window.location.href='index.php";"; 
Jelse{ 
?> 
show_reply() 
<?php 
} 
?>"/><img src="images/t_15_5jpg" width="72" height="23" style="cursor:hand" onclick= "<?php 
/如 果 $_SESSION["unc"] 的 值 为 空 ， 则 不 可 以 进行 项 帖子 的 操作 
if(!isset($_SESSION["unc"])X{ 
echo "javascript:alert(' 请 先 登录 本 站 ,然后 进行 此 操作 ! ');window.location.href='index.php';"; 
Jelse{ 
/否则 将 判断 当前 用 户 的 类 型 ， 如 果 是 管理 员 则 可 以 项 帖 
$sqlu=mysqli_query($conn,"select usertype from tb_user where usernc=".$_SESSION 


[unc] 
Sinfou=mysqli_fetch_array($sqlu); 
9 if($infou["usertype"]==1){ // 如 果 用 户 的 类 型 为 1， 则 有 项 帖 的 权限 
echo "javascript:window.location.href="'settop.php?id=".$infob["id"]."™™"; 
Jelse{ /否则 不 具备 该 权限 
echo "javascript:alert(' 对 不 起 ， 您 不 具备 该 操作 权限 ! ");"; 
} 
} 
ee 
"> 
© <?php 


1/ 判断 当前 用 户 是 否 具有 删除 帖子 的 权限 
iflisset($_SESSION["unc"]))\{ 
/| 条件 为 用 户 不 能 为 空 ， 并 且 是 管理 员 ， 才 具备 删除 帖子 的 权限 
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$sqlu=mysqli_query($conn,"select usertype from tb_user where usernc=". 
$_SESSION["unc"].""™"); 
Sinfou=mysqli_fetch_array($sqlu); 
if($infou["usertype"]==1){ 
?><img src="images/lt_15_10.jpg”onclick="javascript:if(window.confirm(' 您 确定 删除 该 帖 
么 ? ')==true){window.location.href='bbs_delete.php?id=<?php echo S$infob["id"]?>";}" style="cursor:hand"/> 
<?php 





二 代码 贴 十 
@ Sinfo4["usertype"]: 判断 用 户 的 类 型 ， 如 果 值 为 1 是 管理 员 ， 否 则 为 普通 会 员 。 
@ $infob[photo]: 判断 发 布 的 帖子 中 是 否 含有 图 片 ， 如 果 有 则 输出 ， 没 有 则 不 输出 。 
@ stripslashes0: 将 应 用 addcslashes0 函 数 处 理 后 的 字符 串 返 回 原样 。 
@@ 判断 登录 用 户 是 否 具有 顶 帖 的 权限 . 
日 判断 登录 用 户 是 否 具有 删除 帖子 的 权限 。 


~ 
Nm Ls 的 是 该 文件 的 部 分 代码 ， 主 要 讲解 了 该 功能 的 实现 方法 ， 完 整 的 代码 可 以 参考 
本 书 资源 包 中 的 TMNO2\bcty365\bbs lookbbs.php 文件 。 


2.11.5 ”论坛 帖子 发 布 的 实现 过 程 


国 ”论坛 帖 子 发 布 使 用 的 数据 表 : tb_bbs、tb_user 
论坛 帖子 发 布 通过 两 个 文件 来 完成 : 一 个 是 帖子 发 布 信息 的 提交 页 bbs_pubbs.php; 另 一 个 是 对 提 
交 的 数据 进行 处 理 的 retrieve.php 文件 。 该 功能 实现 的 运行 结果 如 图 2.37 所 示 。 





EEalE 3 








图 2.37 帖子 发 布 模块 的 运行 结果 
在 发 布 信息 的 提交 页 中 ， 显 示 当 前 用 户 的 个 人 信息 ， 设 置 添加 数据 表单 元 素 ， 其 中 表单 元 素 的 设 


计 如 表 2.9 所 示 。 
@ 
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表 2.9 发 布 信息 页 中 使 用 的 表单 元 素 





名 称 | 元 素 类 型 重要 属性 费 ” 灸 
form bbs form method="post" action="retrieve.php" enctype="multipart/form-data" 发 帖 表单 
class="inputcss" style="backeround-color:#6EBEC7"> 
<2php 


$sql=mysqli_query($conn,"select * from tb type small order by 
createtime desc"): 

Sinfo=mysqli fetch array($sqD):; 

这 $info 一 包 lse){ 


echo "<option> 暂 无 讨论 区 </option>": 
bbs type Select jelse{ 言 或 者 技 
do{ 术 的 类 别 


?> 
<option value="<?php echo $info['id]:?>"<?php 这 $_GET[Tid] 一 $inforid]) 
{echo "selected=\"selected\"";}?>><?php echo $info['title]:?></option> 
<?php } 
while($info=mysqli fetch array($sql)): 
}?> 

bbs title text class="inputcss" style="background-color:#6EBEC7"> 帖子 标题 
bbs_ head radio value="<?php echo("images/bbsface/face".($i-1).".gif"):?>" 表情 图 
bbs_photo file id="bbs_photo" class="inputcss" style="background-color:#6EBEC7" /> 上 传 图 片 
contentl textarea jd="contentl" class="inputcss" style="background-color:#6EBEC7"> 帖子 内 容 
Submit Submit value=" 提 交 " 提交 表单 


在 retrieve.php 页 中 对 表单 提交 的 数据 进行 处 理 ， 将 数据 存储 到 tb_bbs 表 中 ,并且 更 新 用 户 信息 表 
tb_user 中 pubtimes 字段 的 值 ， 其 中 还 应 用 了 图 片上 传 技 术 ， 将 图 片上 传 到 服务 器 中 指定 的 文件 夹 下 。 
retrieve.php 文件 的 代码 如 下 : 


倒 程 17 代码 位 置 ， 资源 包 \TM\02\bety365\retrieve.php 














<?php 
header("Content-type: text/html; charset=utf-8"); // 设 置 文件 编码 格式 
session_start(); /初始 化 session 变量 
S$title=$_POST[bbs title]; // 获 取 帖 子 的 标题 
$content=$_POST[content11; /获取 帖子 的 内 容 
/判断 提交 的 帖子 主题 和 帖子 内 容 是 否 为 空 "/ 
if($title=="){ 

echo "<script>alert(' 请 输入 帖子 主题 !');history.back();</script>"; 

exit; 
} 


if($content=="™"){ 
echo "<script>alert(' 请 输入 帖子 内 容 ! ');history.back();</script>"; 
exit; 
} 
include_once("conn/conn.php"); /连接 数据 库 


@ 
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date_default_timezone_set("PRC"); 
/根据 $_SESSION["unc"] 的 值 读 取 数 据 库 中 用 户 的 信息 
$sql=mysqli_query($conn,"select * from tb_user where usernc=".$_SESSION["unc"]."”); 


$info=mysqli_fetch_array($sql): /检索 指定 条 件 的 数据 信息 
$userid=$info['id]; // 获 取 用 户 id 
Stypeid=$_POST[bbs_type']; // 接 收 版 块 名 称 
$title=$_POST[bbs title]; /接收 帖子 主题 
$content=$_POST[content11; /接收 帖子 内 容 
$head=$_POST[bbs_head]; /接收 头像 
$createtime=date("Y-m-j H:i:s"); // 获 取 系 统 当前 时 间 
S$lastreplytime=$createtime; // 将 当前 时 间 赋 给 变量 
S$readtimes=0; 


$link=date("YmjHis"); 
@ if($_FILES['bbs_photo"]["name"]==trueX // 上 传 图 片 ， 判 断 文件 是 否 存在 ， 如 果 存 在 则 执行 下 面 的 内 容 
@ S$photo_name=strtolower(stristr($_FILES["bbs_photo"]["name"],".")); /获取 图 片 后 缀 名 , 将 字符 转换 成 
小 写 
if($photo_name!=".gif' && $photo_name!=".jpg" && $photo_namel="jpeg"){ // 判 断 图 片 的 格式 是 否 符合 要 求 
echo "<script>alert( 您 上 传 的 图 片 格式 不 正确 !);history.back();</script>"; 


Jelse{ 
© $paths1=$link.mt_rand(1000000,9999999).$photo_name; // 创 建 图 片 的 名 称 
$photos="./upfile/". $paths1; // 创 建 图 片 的 存储 路 径 
9 move_uploaded file($_FILES["bbs_photo]["tmp_name"],$photos); /将 图 片 存 储 到 指定 的 
文件 夹 下 
// 向 数据 库 添加 数据 


if(mysqli_query($conn,"insert into tb_bbs(userid,typeid ,title,content,createtime,lastreplytime,head, 
readtimes,top,photo) values (".$userid.",".$typeid.",".$title.",".$content.",".$createtime.",".$lastreplytime.", 
".$head.",".$readtimes.",'0''$photos)")}{ 
mysqli_query($conn,"update tb_user set pubtimes=pubtimes+1");// 更 新 tb_user 中 pubtimes 字 
段 的 值 
echo "<script>alert( 新 帖 发 表 成 功 ");history.back();</script>"; 
mysqli_close($conn); 


jelsef 
echo "<script>alert( 新 帖 发 表 失 败 ');history.back();</script>"; 
mysqli_close($conn); 

} 


} 
}else{// 如 果 没 有 提交 图 片 ， 则 执行 下 面 的 内 容 
if(mysqli_query($conn,"insert into tb_bbs(userid,typeid ,title,content,createtime,lastreplytime,head,readtimes,top) 
values 
("$userid.",". $typeid.™,". $title.",”".$content.",”".$createtime.",”".$lastreplytime.",".$head.”,".$readtimes.",'0")")\{ 
mysqli_query($conn,"update tb_user set pubtimes=pubtimes+1"); 
echo "<script>alert( 新 帖 发 表 成 功 !");history.back();</script>"; 
mysqli_close($conn); 


Jelse{ 
echo "<script>alert(' 新 帖 发 表 失 败 !");history.back();</script>"; 
mysqli_close($conn); 
} 
} 
?> 
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< 代码 贴 十 
@S$ FILES[bbs photo']["name"]: $ FILES[] 全 局 变量 ， 获 取 表 单 提交 文件 的 原始 名 称 。 
@ strtolower0: 将 指定 的 字符 转换 为 小 写字 母 。 
stristr0: 获取 指定 字符 囊 ( A) 在 另 一 个 字符 囊 (B) 中 首次 出 现 的 位 置 到 (B) 字符 囊 末 尾 的 所 有 字符 囊 。 该 函 
数 如 果 执 行 成 功 则 返回 剩余 的 字符 囊 ， 否 则 将 返回 false-。 
@@ mt rand0: 生成 一 个 随机 数 ， 用 于 上 传 文件 的 名 称 。 
@ move_uploaded file(): 将 指定 的 文件 上 传 到 指定 的 文件 夹 下 。 


2.11.6 ”论坛 帖子 回复 的 实现 过 程 


国 论坛 帖子 回复 使 用 的 数据 表 : tb_bbs、tb_user、tb_reply 
回复 论坛 中 的 帖子 ， 必 须 是 以 会 员 或 者 管理 员 的 身份 进行 登录 ， 和 否则 不 能 进行 帖子 的 回复 操作 ， 
其 运行 结果 如 图 2.38 所 示 。 





生 册 时 间 : aotT-15-24 cy 46:14 各 


图 2.38 论坛 帖子 回复 的 运行 结果 


论坛 帖子 回复 功能 的 实现 主要 通过 bbs_lookbbs.php 和 savereplyphp 两 个 文件 。 其 中 应 用 JavaScript 
脚本 对 回复 帖子 的 文本 框 进行 输出 和 隐藏 的 控制 。 在 bbs_lookbbs.php 文件 中 ， 帖 子 回复 使 用 的 表单 元 
素 如 表 2.10 所 示 。 


表 2.10 论坛 帖子 回复 中 的 重要 表单 元 素 








名 称 | 元 素 类 型 重要 属性 含义 
form reply form | method="post" action="savereply.php" enctype="multipart/form-data"> | 回复 表单 
reply_title text class="inputcss" id="reply_title" 回复 帖子 主题 
bbsid hidden value="<?php echo $infob["id"]:?>" 对 应 帖子 的 ID 
bbs head Tadio value="<?php echo("images/bbsface/face".($i-1).".gif’):?>" 表情 图 
bbs_photo file id="bbs_photo" class="inputcss" 上 传 图 片 
contentl textarea id="content1" 回复 帖子 内 容 
Submit submit value=" 提 交 " 提交 表单 











在 帖子 回复 表单 bbs_lookbbs.php 页 中 ， 首 先 判断 登录 用 户 是 否 具有 回复 的 权限 ， 然 后 根据 提交 的 


© 


第 2 章 BCTY365 网 上 社区 (Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 
值 展 开 回复 表单 的 文本 框 ， 在 文本 框 中 输入 回复 的 主题 和 内 容 ， 最 后 将 数据 提交 到 表单 处 理 页 
savereply.php 中 。bbs_lookbbs.php 的 主要 代码 如 下 : 
倒 程 18 ”代码 位 置 ， 资源 包 \TM\02\bcty365\bbs_lookbbs.php 


<script language="javascript"> 





// 设 计 回复 帖子 表格 的 输出 方式 
function show_reply()f /定义 一 个 函数 
if(reply_bbs1.style.display=="") /| 判断 当 display 的 值 为 空 时 
reply_bbs1.style.display="none"; /1 不 输出 表格 
} 
else if(reply_bbs1.style.display=="none") 1/ 判断 当 display 的 值 为 none 时 
reply_bbs1.style.display="; /输出 表格 
} 
</script> 
< 一 一 一 一 一 一 一 一 一 一 一 一 判断 登录 用 户 是 否 具有 回复 的 权限 一 一 一 一 一 一 一 一 一 一 > 
<img src="images/lt_15_9.jpg" width="72" height="23" id="button_show_bbs" ="cursor:hand" onClick="<?php 


if(!lisset($_SESSION["unc"])X{ 
echo "javascriptalert( 请 先 登录 本 站 ， 然 后 回复 帖子 ! ');window.location.href='index.php""; 


jelsef 


?> 
show_reply() 
<?php 
} 


?>"/> 
表单 处 理 页 savereplyphp 将 表单 提交 的 数据 存储 到 指定 的 数据 库 中 ， 其 实现 的 方法 与 论坛 发 布 中 
的 表单 处 理 技术 是 相同 的 ， 有 关 该 技术 的 详细 讲解 请 参考 2.11.5 节 ， 这 里 不 再 獒 述 。 


2.11.7 ”单元 测试 


在 测试 网 上 社区 的 论坛 模块 时 ， 发 现 发 帖 时 上 传 的 图 片 不 能 够 正常 显示 ， 运 行 结果 中 出 现 错 误 提 
示 ， 如 图 2.39 所 示 。 分 析 错 误 原 因 ， 在 图 片上 传 成 功 后 ， 没 能 正确 地 读 取 数据 库 中 指定 图 片 的 路 径 。 











图 2.39 程序 运行 错误 结果 图 


在 bbs_lookbbs.php 页 面 中 查看 获取 的 图 片 路 径 是 否 正 确 。 发 现在 读 取 数据 库 中 图 片 路 径 的 代码 段 
中 ,使 用 了 错误 的 字段 名 称 ， 数据库 中 图 片 路 径 存 储 使 用 的 字段 名 是 photo， 而 在 程序 代码 段 中 使 用 的 


却 是 photos。 错 误 代码 如 下 : 
@ 
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<?php 


if($infob[ photo"]!="™"X{ // 判 断 是 否 存在 图 片 
$photos=substr($infob['photos"],2,70); // 获 取 图 片 存储 的 路 径 
echo (stripslashes($infob["content])); /| 输出 帖子 的 内 容 
echo "<img src=\"$photos\">"; /| 输出 图 片 

}else{ 
echo (stripslashes($infob["content"])); /| 输出 帖子 内 容 


} 


?> 


将 代码 段 中 的 字段 名 进行 修改 ， 然 后 重新 运行 程序 ， 图 片 正常 显示 。 





2.12 后 台 首 页 设计 


作为 一 个 完整 的 网 上 社区 系统 ， 要 想 能 够 及 时 地 对 网 站 进行 管理 和 维护 ， 必 须 具 有 一 个 强大 的 后 
台 管理 系统 ， 对 网 上 社区 系统 中 的 数据 进行 更 新 和 维护 。 


2.12.1 后 台 首 页 概述 


网 上 社区 系统 的 后 台 管理 采用 的 是 一 种 简单 的 框架 结构 ， 通 过 switch 语句 来 实现 。 其 包括 如 下 具 
体内 容 。 
软件 试用 管理 : 包括 软件 试用 产品 的 添加 和 删除 。 
编程 词典 管理 : 包括 编程 词典 版 本 的 添加 、 删 除 和 编程 词典 内 容 的 添加 和 删除 。 
在 线 订 购 管理 : 主要 用 于 管理 用 户 提交 的 订单 。 
软件 升级 管理 : 包括 升级 包 的 添加 、 删 除 和 序列 号 的 添加 和 删除 。 
站 内 公告 管理 : 主要 用 于 添加 和 删除 站 内 公告 。 
技术 支持 管理 : 主要 用 于 添加 常见 问题 和 删除 常见 问题 ， 以 及 对 客户 反馈 信息 进行 管理 。 
本 案例 中 提供 的 后 台 首页 如 图 2.40 所 示 。 该 页 面 在 本 书 资源 包 中 的 路 径 为 \TM\02\bcty365\admin\ 
defaultphp 。 


加 回回 罗 罗 加 
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图 2.40 BCTY365 网 上 社区 系统 后 台 首 页 
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2.12.2 后 台 首 页 技术 分 析 


网 上 社区 后 台 首 页 的 设计 主要 应 用 switch 语句 和 include 包含 语句 ， 其 实现 的 原理 是 : 应 用 switch 
语句 ， 根 据 超 链接 中 传递 的 变量 值 进 行 判断 ， 根 据 不 同 的 变量 值 应 用 include 调用 不 同 的 子 文件 。 该 技 
术 的 实现 流程 如 图 2.41 所 示 。 












应 用 站 语句 判断 变 
量 提交 的 值 
要 户 服务 宁 心 纪 引 0 


图 2.41 网 上 社区 后 台 首页 设计 流程 
为 了 能 够 更 好 地 理解 这 个 技术 ， 先 来 了 解 一 下 switch 语句 。 该 语句 的 格式 如 下 : 


(加 | 应 用 inciude 根 据 不 同 的 
值 输出 不 同 的 子 文件 


Creriht 201T 区 ) 六 村 者 妥 科技 有 限 公 避 











Switch(exprjf /expr 条 件 为 变量 名 称 
case expr1: licase 后 的 expr1 为 变量 的 值 
statement1; // 冒 号 (:) 后 的 是 符合 该 条 件 时 要 执行 的 部 分 
break; /应 用 break 来 跳 离 循环 体 
Case expr2: 
statement2; 
break; 
default: 
statementN; 


break; 


} 





参数 expr 是 表达 式 的 值 ， 即 switch 语句 的 条 件 变量 的 名 称 ， 参 数 exprl 放置 于 case 语句 之 后 ， 是 
要 与 条 件 变 量 expr 进行 匹配 的 值 中 的 一 个 ;statementl 是 在 参数 exprl 的 值 与 条 件 变 量 expr 的 值 相 匹配 
时 执行 的 代码 ，break 语句 实现 终止 语句 的 执行 ， 即 当 语 句 在 执行 过 程 中 ， 遇 到 break 就 停止 执行 ， 跳 出 
循环 体 ; default 是 case 的 一 个 特例 ， 匹 配 了 任何 其 他 case 都 不 匹配 的 情况 ， 并 且 是 最 后 一 条 case 语句 。 

通过 switch 和 include 语句 来 实现 后 台 管理 功能 的 设计 是 一 个 很 好 的 方法 ， 不 但 实现 过 程 简 单 ， 而 
且 操 作 也 非常 灵活 。 关 键 代 码 如 下 : 


倒 竹 19 ”代码 位 置 : 资源 包 \TM\02\bcty365\admin\wzdh.php 


<?php 
switch(isset($_GET[htgl])?$_GET[htgl]:")( /根据 变量 提交 的 不 同 值 
case "添加 编程 词典 版 本 ": // 判 断 与 变量 提交 的 值 是 否 相同 
include("addbb.php"); // 如 果 值 相同 ， 则 调用 指定 的 文件 
break; // 并 且 跳 出 本 次 循环 


case "编辑 编程 词典 版 本 "- 
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include("editbd.php"); 
break;: 
Ee // 部 分 代码 省 略 
case ™: // 当 变量 的 值 为 空 时 
include("edittell.php"); // 调 用 该 文件 
break' 
?> 





2.12.3 ”后 台 首页 的 实现 过 程 


国 后台 首页 使 用 的 数据 表 : tb bb 
在 后 台 首页 的 设计 过 程 中 ， 以 switch 语句 为 基础 ， 架 设 整 个 后 台 管 理 功能 的 框架 结构 ; 充分 发 挥 


include 包含 语句 的 作用 ， 调 用 不 同 的 文件 执行 不 同 的 管理 操作 ;应 用 JavaScript 脚本 来 控制 栏目 列表 
的 输出 和 隐藏 。 


控制 栏目 列表 的 输出 和 隐藏 在 menu.php 文件 中 进行 ， 首 先 定义 一 个 函数 changeO 用 于 控制 表格 的 


输出 和 隐藏 ， 然 后 在 表格 中 应 用 onclick 事件 传递 不 同 的 值 到 自 定义 函数 change0， 最 后 根据 不 同 的 值 
显示 不 同 的 内 容 。 关 键 代码 如 下 : 


倒 程 20 代码 位 置 ， 资源 包 \TM\02\bcty365\admin\menu.php 





<script language="javascript"> 


// 通 过 脚本 语言 控制 文本 框 的 伸展 和 收缩 
function change(x,yjf /定义 一 个 函数 
if(x.style.display=="none")}{ 1/ 判断 当 样式 的 值 为 none 时 
x.style.display="™"; // 输 出 样式 的 值 为 空 
二 
else if(x.style.display=="" /| 判断 当 样 式 的 值 为 空 时 
x.style.display="none"; // 输 出 样式 的 值 为 none 
y.background="images/bg_16_11.jpg"; /| 输出 背景 图 片 
} 
= 
</script> 


<table width="175" height="28" border="0" align="center" cellpadding="0" cellspacing="4" 
onclick="change(tz1,img_tz1)" style="cursor:hand"> 

<tr> 

<td background="images/bg_16_11.jpg" id="img_tz1" class="a4"><div align="left"><img 

src="images/bg_16_21.jpg">&nbsp; 编 程 词典 管理 </div></td> 

</tr> 
</table> 
<table name="tz1" id="tz1" width="170" height="40" border="0" align="center" cellpadding="0" cellspacing="0" 
<?php 

// 根 据 变量 的 值 选择 执行 的 内 容 ， 当 变量 的 值 不 为 真 时 ， 隐 藏 该 表格 
ifisset($_GET[htgl]) && !$_GET[htgl]==" 添 加 编程 词典 版 本 "|l$_GET[htgl]==" 编 辑 编程 词典 版 本 " 
ll$_GET[htgl]==" 添 加 编程 词典 " ||$_GETThtgl]==" 编 辑 编程 词典 ")){ 
?> 
style="display:none”" 
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<?php 六 
> 
<tr> 

<td width="40" height="24" background="images/bg_16_16.jpg">&nbsp;</td> 

<td width="114" background="images/bg_16_16.jpg"><div align="left"><a href="default.php?htgl= 添 加 编程 
词典 版 本 "> 添加 编程 词典 版 本 </a></div></td> 

</tr> 

</table> 





| 
bcty365\admin\ 文 件 夹 下 的 相关 文件 。 


2.13 ”编程 词典 管理 模块 设计 








本 模块 的 功能 是 对 网 站 中 的 编程 词典 进行 管理 ， 包 括 添加 编程 词典 版 本 、 编 辑 编程 词典 版 本 、 添 
加 编程 词典 和 编辑 编程 词典 。 


2.13.1 编程 词典 管理 模块 概述 


本 模块 的 主要 功能 是 管理 网 站 中 在 线 出 售 的 编程 词典 软件 ， 实 现 对 编程 词典 软件 的 及 时 更 新 和 维 
护 ， 其 管理 的 内 容 主要 包括 添加 和 编辑 编程 词典 的 版 本 ， 添 加 和 编辑 编程 词典 的 详细 信息 。 在 添加 编 
程 词典 时 ， 包 括 名 称 、 版 权 、 图 片 、 类 别 、 内 容 简介 和 不 同 版 本 的 共同 点 ;编辑 编程 词典 包括 版 本 、 
价格 、 简 介 、 功 能 和 服务 ， 其 中 每 一 个 编程 词典 软件 只 可 以 编辑 一 次 ， 不 可 以 进行 重复 编辑 ， 如 果 要 
重新 编辑 ， 就 必须 将 已 经 编辑 过 的 信息 删除 。 


2.13.2 ”编程 词典 管理 模块 技术 分 析 


在 编程 词典 管理 模块 中 ， 应 用 到 图 片上 传 技术 ， 通 过 该 技术 将 编程 词典 的 界面 效果 上 传 到 服务 器 
的 指定 文件 夹 下 。 该 技术 主要 通过 move_uploaded_file0 函 数 来 实现 ， 其 中 还 应 用 到 is_dir0、mkdir0 函 
数 ， 判 断 指定 的 文件 夹 是 否 存 在 和 创建 文件 来， 还 有 mt rand0、strstr0 函 数 和 $_FILES[] 全 局 变量 。 为 
了 更 好 地 理解 和 掌握 图 片上 传 处 理 技术 ， 这 里 以 编程 词典 模块 中 的 savebccd.php 文件 为 例 进行 讲解 。 

首先 应 用 is_dir0 函 数 判 断 在 服务 器 中 是 否 存在 指定 的 文件 夹 ， 如 果 不 存在 ， 则 应 用 mkdir0 函 数 创 
建 一 个 新 的 文件 夹 。 

然后 应 用 $_FILES[] 全 局 变量 获取 图 片 名 ， 应 用 strstr0 函 数 获取 图 片 文件 的 后 级 名 ， 为 了 避免 出 现 
同名 文件 覆盖 ， 这 里 应 用 系统 的 当前 时 间 和 mtrand0 函 数 获取 的 一 个 7 位 随机 数字 作为 图 片 的 名 称 。 

最 后 确定 图 片 在 服务 器 中 存储 的 路 径 ， 将 图 片上 传 到 指定 的 文件 夹 下 。 而 数据 库 中 存储 的 数据 是 
图 片 在 服务 器 中 的 路 径 ， 当 需要 输出 图 片 时 ， 只 需要 获取 到 数据 库 中 图 片 的 路 径 即 可 。savebccd.php 


文件 的 代码 如 下 : 
@ 
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倒 程 21 代码 位 置 资源 包 \TM\02\bcty365\admin\savebccd.php 





<?php 

header("Content-type: text/html; charset=utf-8"); /设置 文件 编码 格式 
include_once("../conn/conn.php"); /连接 数据 库 
$bccdname=$_POST[bccdname"]; /获取 POST 方法 提交 的 值 


$owner=$_POST[owner]; 
$typeid=$_POST[typeid]: 
$content=$_POST[content]; 
$samepart=$_POST[samepart]; 
date_default_timezone_set("PRC"); 


$addtime=date("Y-m-j H:i:s"); // 获 取 当 前 时 间 

@ iflis_dir("/bccdimages")==false)f 1/ 判断 指定 的 文件 是 否 存 在 

@ mkdir("./bccdimages"); /如果 不 存在 ， 则 创建 一 个 新 的 文件 夹 
} 

S$link=date("YmjHis"); // 获 取 当 前 时 间 


// 为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 
// 其 中 使 用 $_FILES 获取 表单 中 真实 的 名 称 ， 使 用 strstr() 函 数 获取 文件 的 后 缀 
© $path=$link.mt_rand(1000000,9999999).strstr($_FILES["imageaddress"]["name"],"."); 


S$address="./bccdimages/". $path; /定义 文件 上 传 的 路 径 

@ move_uploaded_file($_FILES["imageaddress"]["tmp_name"],$address);// 将 文件 上 传 到 指定 的 文件 中 
S$imageaddress="./admin/bccdimages/".$path; /| 获取 上 传 文件 在 服务 器 中 的 存储 路 径 

// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


$query=mysqli_query($conn,"insert into tb_bccd(bccdname,ownertypeid,content,samepart,imageaddress,addtime) 
values('$bccdname','$owner','$typeid','$content','$samepart','$imageaddress','$addtime')"); 
if($query==trueX{ 
echo "<script>alert(' 编 程 词典 添加 成 功 ! ');history.back();</script>"; 
jelsef 
echo "<script>alert(' 编 程 词典 添加 失败 ! ');history.back();</script>"; 
} 


?> 





Ah 代码 贴 二 
@is dir0: 判断 指定 的 文件 夹 是 否 存 在 ， 如 果 存 在 则 返回 tue， 否 则 返回 false。 
@ mkdir0: 创建 一 个 新 的 文件 夹 。 
上 @ mt_rand(): 根 据 提供 的 参数 min 和 max 生成 随机 数 ,如果 没有 提供 可 选 参数 min 和 max, 则 返回 0 到 RAND MAX 
之 间 的 伪 随机 数 。 
strstr0: 获取 一 个 指定 字符 串 在 另 一 个 字符 串 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字符 串 。 如 果 执 行 成 功 ， 则 返回 
剩余 字符 囊 ( 存在 相 匹配 的 字符 ); 如 果 没有 找到 相 匹配 的 字符 ， 则 返回 false。 
$_FILES[]: 全 局 变量 , 获取 所 有 上 传 文件 的 信息 .该 全 局 变量 还 可 以 获取 到 其 他 的 值 ,其 中 $_FILES["imageaddress'] 
[mame'] 获 取 的 是 客户 端 机 器 文件 的 原名 称 ; $_FILES['imageaddress'['size'] 获 取 已 上 传 文件 的 大 小 , 单位 为 字 节 ; $_FILES 
[imageaddress']['tmp_name'] 获 取 文 件 被 上 传 后 在 服务 端 存储 的 临时 文件 名 ; $ FILES[imageaddress']['error] 获 取 和 该 文件 
上 传 相 关 的 错误 代码 。 
@ move_ uploaded file(string filename.string destination): 应 用 POST 方法 实现 文件 的 上 传 ， 参 数 filename 指定 要 上 
传 的 文件 地 址 ; 参数 destination 指定 文件 上 传 到 服务 器 后 的 存储 目录 及 名 称 。 


汗 诗 
全 注意 应 用 POST 方法 上 传 图 片 文件 时 ， 应 当 在 上 传 表单 的 <form> 标 记 中 添加 内 容 “enctype= 
"multipart/form-data"” 。 


全 
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2.13.3 ”添加 编程 词典 的 实现 过 程 


国 添加 编程 词典 使 用 的 数据 表 : tb bccd、tb type 
添加 编程 词典 的 功能 是 向 数据 库 中 添加 编程 词典 的 详细 信息 ， 包 括 编程 词典 的 名 称 、 版 权 、 图 片 、 
类 别 、 内 容 简介 和 不 同 版 本 的 共同 点 。 其 运行 结果 如 图 2.42 所 示 。 
二 

















这 得 癌 直 名 标 : ET 

版 权 所 有 : 周 补 吉明 巨 科技 有 限 公司 

邵 片 癌 径 + 2017! 

所 属 关 别 : 

人: 

他 对 不 同 ] 扳 本 ， 基 中 并且 3 不 同 允 号 程 启 的 jie 
Te 
到。 到 





图 2.42 添加 编程 词典 模块 的 运行 结果 


添加 编程 词典 信息 模块 主要 通过 addbccd.php 和 savebccd.php 文件 来 完成 ， 其 中 在 addbccd.php 文 
件 中 主要 是 设计 表单 元 素 ， 而 savebccd.php 文件 主要 是 对 表单 中 提交 的 数据 进行 处 理 。addbccd.php 文 
件 中 使 用 的 表单 元 素 如 表 2.11 所 示 。 


表 2.11 添加 编程 词典 页 中 使 用 的 重要 表单 元 素 











名 称 | 元 素 类 型 重要 属性 含 义 

i po Doe Pot action="savebccd.php”onSubmit="returm chkinput(this)" 编程 词典 表单 

enctype="multipart/form-data"> 
bccdname text Siz Brey" 编程 词典 名 称 
OWneT text i 卉 版 权 所 有 者 
imageaddress file Size="25" class="txt_grey" 界面 图 片 

<?php include_once("../conn/conn php"): 

S$sql=mysqli_query($conn."select * from tb_type order by createtime desc"); 

Sinfo=mysqli fetch array($sqD): 

这 $info 一 false){ echo "<option > 暂 无 类 别 </option>"; 选择 编程 词典 的 

typeid select 

jelse{ 版 本 

dof 

echo "<option value=".$info['id].">".$info[typename']."</option>": } 

while($info=mysqli fetch array($sqD):} > 
content textarea TOWs="10" cols="65" class="textarea" 内 容 简 介 
samepart textarea TOWs="10" cols="65" class="textarea" 不 同 版 本 的 特点 
Submit submit value=" 添 加 " class="btn_grey" 提交 表单 
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savebccd.php 文件 实现 对 表单 中 提交 的 数据 进行 处 理 ， 首 先 通过 $_ POST 获取 表单 中 提交 的 数据 ， 
然后 判断 指定 的 文件 夹 是 否 存 在 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 关 键 代码 如 下 : 


倒 程 22 ”代码 位 置 ， 资源 包 \TM\02\bcty365\admin\savebccd.php 


<?php 

header("Content-type: text/html; charset=utf-8"); /设置 文件 编码 格式 
include_once("../conn/conn.php"); /连接 数据 库 
$bccdname=$_POST[bccdname']; /获取 POST 方法 提交 的 值 


$owner=$_POST[owner]; 
$typeid=$_POST[typeid]; 
$content=$_POST['content]:; 
$samepart=$_POST[samepart]; 
date_default_timezone_set("PRC"); 


S$addtime=date("Y-m-j Hii:s"); // 获 取 当 前 时 间 

@ iflis_dir("/bccdimages")==false)f // 判 断 指定 的 文件 是 否 存 在 

@ mkdir("./bccdimages"); // 如 果 不 存在 ， 则 创建 一 个 新 的 文件 夹 
} 

Slink=date("YmjHis"); // 获 取 当 前 时 间 

// 为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 ， 其 中 使 用 $_FILES 获取 表单 中 真实 的 名 称 ， 使 用 
strstr() 函 数 获取 文件 的 后 组 

© $path=$link.mt_rand(1000000,9999999).strstr($_FILES["imageaddress"]["name"],"."); 
S$address="./bccdimages/". $path; // 定 义 文件 上 传 的 路 径 
move_uploaded_file($_FILES["imageaddress"]["tmp_name"],$address);，// 将 文件 上 传 到 指定 的 文件 中 
S$imageaddress="./admin/bccdimages/".$path; /| 获取 上 传 文件 在 服务 器 中 的 存储 路 径 
// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


$query=mysqli_query($conn,"insert into tb_bccd(bccdname,ownertypeid,content,samepartimageaddress,addtime) 
values('$bccdname','$owner','$typeid','$content','$samepart','$imageaddress','$addtime')"); 
if($query==trueX{ 
echo "<script>alert(' 编 程 词典 添加 成 功 !"");history.back();</script>"; 
Jelse{ 
echo "<script>alert(' 编 程 词典 添加 失败 !");history.back();</script>"; 
} 


?> 


< 代码 贴 二 
@is dir0: 判断 指定 的 文件 是 否 存在 。 
@ mkdir0: 创建 一 个 新 的 文件 夹 。 
目 mt rand0: 获取 随机 数字 。 
strstr0: 获取 一 个 指定 字符 囊 在 另 一 个 字符 囊 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字符 囊 。 如 果 执行 成 功 ， 则 返回 
剩余 字符 串 (存在 相 匹配 的 字符 ); 如 果 没 有 找到 相 匹配 的 字符 ， 则 返回 false. 


2.13.4 ”编辑 编程 词典 的 实现 过 程 


国 编辑 编程 词典 使 用 的 数据 表 : tb bccd、tb bb、tb bbqb 
在 完成 对 编程 词典 信息 的 添加 后 ， 接 下 来 就 可 以 对 编程 词典 的 版 本 信息 进行 编辑 ， 主 要 添加 版 本 
人 信息、 价格、 简介 、 功 能 和 推出 的 服务 。 该 模块 的 运行 结果 如 图 2.43 所 示 。 
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图 2.43 编辑 编程 词典 模块 的 运行 结果 


该 功能 的 实现 同样 通过 两 个 文件 : 一 个 是 提交 表单 的 文件 editbccd.php; 另 一 个 是 处 理 表单 提交 数 
据 的 文件 sacvbccdbb.php。 提 交 表 单 文件 editbccd.php 中 使 用 的 表单 元 素 如 表 2.12 所 示 。 


名 称 


forml 


bccdname 


bccdid 


bbid 


元 素 类 型 


form 


text 


hidden 


select 


表 2.12 编辑 编程 词典 页 中 使 用 的 重要 表单 元 素 


重要 属性 
method="post" action="savebccdbb.php" onSubmit="return 
chkinput(this)"> 
class="txt_grey" disabled="disabled" value=" 
<2php 
$sql4=mysqli_query($conn."select bccdname from tb bccd where 
jd=".$_ GET[bccdid].""): 
Sinfo4=mysqli_fetch_array($sql4): 
echo unhtml($info4['bccdname']): ?> 
value="<?php echo $ GET[bccdid]:? 
<?php $sql3=mysqli query($conn."select * from tb_bb order 
by 
createtime desc "): 
Sinfo3=mysqli_ fetch array($sql3): 
if($info3 一 false){ echo “<option> 暂 无 版 本 信息 </option>"; 
Jelse{ 
do{f ?> 
<option value="<?php echo $info3[id]:?>"><?php echo 
unhtml($info3[bbname']):?></option> 
<?php }while($info3=mysqli fetch array($sql3)): }?> 





含 义 
编辑 编程 词典 表单 


编程 词典 名 称 , 这 里 设置 
了 文本 的 只 读 属性 


编程 词典 的 ID 


选择 编程 词典 的 版 本 





Submit 





Submit 





value=" 添 加 " class="btn_egrey" 





提交 表单 
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sacvbccdbb.php 文件 对 表单 提交 的 数据 进行 处 理 , 首先 获取 表单 中 提交 的 数据 , 然后 判断 指定 的 版 
本 是 否 已 经 被 添加 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 代 码 如 下 : 


倒 程 23 ”代码 位 置 ， 资源 包 \TM\02\bcty365\admin\savebccdbb.php 


<?php 
header("Content-type: text/html; charset=utf-8"); /设置 文件 编码 格式 
$bccdid=$_POST[bccdid]; /获取 表单 中 提交 的 数据 
$bbid=$_POST[bbid]; // 获 取 编 程 词典 ID 
$price=$_POST[price]; /获取 编程 词典 单价 
$content=$_POST[content]; // 获 取 编 程 词典 内 容 
$gn=$_POST[gn'; // 获 取 编 程 词典 功能 
$fw=$_POST[fw]; /获取 编程 词典 服务 
include_once("../conn/conn.php"); /| 连接 数据 库 文 件 
// 判 断 提交 的 编程 词典 是 否 已 经 被 添加 
$sql=mysqli_query($conn,"select id from tb_bbqb where bccdid=".$bccdid."”); 
S$info=mysqli_fetch_array($sql); /检索 指定 编程 词典 的 ID 
if($info!=false\{ // 如 果 检 索 值 为 假 ， 则 弹出 提示 
echo "<script>alert(' 该 版 编程 词典 已 经 添加 ! ');history.back();</script>"; 
exit; 


} 

$query=mysqli_query($conn,"insert into tb_bbqb(bccdid,bbid,price,content,gn,fw) values('$bccdid','$bbid','$price’, 
'$content','$gn','$fWw)"); // 将 表单 中 提交 的 数据 存储 到 数据 库 中 

// 更 新 编程 词典 的 价格 

$querys=mysqli_query($conn,"update tb_bccd set bbid='$bbid',price='$price' where id=".$bccdid.""); 

echo mysqli_error(); 


if($query==true and $querys==true}{ /| 如果 添加 和 更 新 操作 为 真 ， 则 弹出 提示 
echo "<script>alert(' 版 本 信息 添加 成 功 ! ');history.back();</script>"; 
}yelse{ // 如 果 添 加 和 更 新 操作 为 假 ， 则 弹出 提示 


echo "<script>alert( 版 本 信息 添加 失败 ! ');history.back();</script>"; 
} 


?> 








2.14 软件 升级 管理 模块 设计 


IE 
2.14.1 软件 升级 管理 模块 概述 


软件 升级 管理 模块 实现 对 软件 升级 包 的 管理 ， 其 具体 的 功能 包括 添加 升级 包 、 编 辑 升 级 包 、 添 加 
序列 号 和 编辑 序列 号 。 软 件 升 级 管理 模块 中 的 添加 升级 包 和 添加 序列 号 是 一 一 对 应 的 ， 其 中 根据 所 属 
的 类 别 和 版 本 来 确定 升级 包 对 应 的 序列 号 ， 每 一 个 版 本 和 类 别 的 升级 包 对 应 一 个 序列 号 。 


2.14.2 ”软件 升级 管理 模块 技术 分 析 


在 软件 升级 包 管理 模块 中 ， 应 用 到 一 个 动态 输出 下 拉 列 表 框 中 值 的 技术 。 下 面 就 来 讲解 一 下 该 技 


@ 
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术 是 如 何 实现 的 ， 在 讲解 该 技术 之 前 ， 先 来 了 解 下 拉 列 表 框 的 基本 结构 。 


<select name="select"><!--name 指定 该 下 拉 列 表 框 的 名 称 --> 
<!--selected 设置 下 拉 列 表 框 的 默认 值 ， 默 认 值 为 PHP--> 
<option selected="selected">PHP</option> 
<!--value 指定 的 mysql 是 下 拉 列 表 框 传递 的 值 ，MYSQL 为 显示 的 内 容 -> 
<option value="mysql">MYSQL</option> 
</select> 


所 谓 动态 输出 下 拉 列 表 框 中 的 值 就 是 从 数据 库 中 读 取 数 据 , 将 获取 到 的 数据 输出 到 下 拉 列 表 框 中 ， 
而 不 是 直接 在 下 拉 列 表 框 中 设置 某 个 固定 的 值 。 这 里 以 软件 升级 管理 模块 addsjb.php 文件 中 的 所 属 类 
别 下 拉 列 表 框 为 例 进 行 讲解 ， 其 中 设置 下 拉 列 表 框 的 名 称 为 typeid， 默认 值 为 “请 选择 ”value 的 值 是 
从 数据 库 中 获取 的 ID 值 ， 显 示 的 内 容 为 从 数据 库 中 获取 的 类 型 名 称 。 动 态 输 出 下 拉 列 表 框 中 的 值 使 用 
的 关键 代码 如 下 : 


倒 程 24 代码 位 置 ， 资源 包 \TM\02\bcty365\admin\addsjb.php 
< 一 一 一 一 一 一 一 一 一 一 设置 下 拉 列 表 框 的 名 称 为 typeid 一 一 一 一 一 一 一 一 -> 
<select name="typeid" class="txt_grey"> 

<option value=" selected="selected"> 请 选择 </option> 





< -一 一 一 一 一 一 一 一 设置 下 拉 列 表 框 的 名 称 为 typeid 一 -一 -一 一 一 一 -> 
<?php 

include_once("../conn/conn.php"); /| 连接 数据 库 

// 从 数据 库 中 读 取 编程 词典 类 型 的 数据 


$sql=mysqli_query($conn,"select * from tb_type order by createtime desc"); 
Sinfo=mysqli_fetch_array($sql); 
if($info==false}{ 
echo "<option > 暂 无 类 别 </option>"; 
jelsef 
dof /应 用 do…while 循环 语句 输出 类 型 的 ID 和 类 型 的 名 称 
echo "<option value=".$info['id"].">".$info['typename']."</option>"; 


} 
while($info=mysqli_fetch_array($sql)); /do…while 循环 语句 结束 
} 


VY 
</select> 


下 拉 列 表 框 不 但 可 以 动态 输出 数据 库 中 某 个 字段 的 数据 ， 而 且 可 以 输出 数组 中 的 数据 。 下 面 就 实 
现 一 个 在 下 拉 列 表 框 中 动态 输出 数组 中 数据 的 功能 ， 首 先 创建 一 个 下 拉 列 表 框 ， 然 后 设置 下 拉 列 表 框 
的 值 ， 从 数组 中 读 取 数据 ， 应 用 for 循环 语句 进行 输出 。 代 码 如 下 : 





一 一 设置 下 拉 列 表 框 的 名 称 为 select 一 -一 -一 -一 一 一 一 一 一 一 
<select name="select" size="1"> 
<?php 
$string="ASP,PHP,JSP,.NET,DEL,VB,VC"; // 定 义 一 个 字符 串 
Ssrtings=split(",", $string); /对 字符 串 进行 分 割 
S$count=count($srtings); // 获 取 数 组 中 元 素 的 数量 
for($i=0;$i<$count:$i++X{ // 根 据 数组 中 元 素 的 数量 进行 循环 输出 





_ 国 
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$result=$srtings[$i]; /定义 变量 ， 获 取 数 组 中 指定 的 元 素 
echo "<option>$result</option>":} /将 数组 中 的 元 素 输出 到 下 拉 列 表 框 中 
和 
</select> 





2.14.3 ”软件 升级 包 上 传 的 实现 过 程 


国 ”软件 升级 包 上 传 使 用 的 数据 表 : tb bb、tb type、tb_sjxz 


软件 升级 包 上 传 在 添加 升级 包 模块 中 实现 ， 通 过 一 个 文件 域 文本 框 将 升级 包 提交 到 服务 器 中 指定 


的 文件 下 , 并 且 将 该 文件 在 服务 器 中 的 路 径 存储 到 数据 库 




















中 ， 便 于 在 前 台 实现 对 软件 升级 包 的 下 载 。 其 运行 结果 如 
2.44 所 示 。 | 
在 本 模块 中 通过 addsjb.php 文件 来 提交 升级 包 的 信 ee 
息 ， 通 过 savesj.php 文件 来 对 表单 提交 的 数据 进行 处 理 。 aa 
其 中 在 将 升级 包 上 传 到 服务 器 的 指定 文件 夹 的 过 程 中 , 主 到 到 
要 应 用 的 是 move_uploaded_file0 函 数 。 在 savesj.php 文件 2.44 ”软件 升级 包 上 传 的 运行 结果 


中 , 首先 获取 表单 提交 的 数据 , 然后 判断 服务 器 中 是 否 存 
在 指定 的 文件 ， 最 后 应 用 move_uploaded_file0 函 数 将 升级 包 上 传 到 指定 的 文件 夹 下 ， 并 且 将 数据 存储 
到 指定 的 数据 表 中 。 程 序 代 码 如 下 : 


倒 程 25 代码 位 置 ， 资源 包 \TM\02\becty365\admin\savesij.php 





<?php 

header("Content-type: text/html; charset=utf8"); /设置 文件 编码 格式 

$name=$_POST[name]; // 获 取 表 单 提交 的 数据 

S$typeid=$_POST[typeid']: // 获 取 表 单 提交 的 数据 

$content=$_POST[content]; // 获 取 表 单 提 交 的 数据 

date_default_timezone_set("PRC"); // 设 置 时 区 

$addtime=date("Y-m-j Hii:s"); // 定 义 时 间 变 量 

$bbid=$_POST[bbid]; // 获 取 表 单 提 交 的 数据 

if(is_dir("./sjxz")==falseX{ // 判 断 指定 的 文件 夹 是 否 存在 
mkdir("./sjxz"); // 如 果 指 定 的 文件 夹 不 存在 ， 则 创建 一 个 指定 的 文件 夹 

} 

S$link=date("YmjHis"); // 获 取 一 个 时 间 

$path=$link.mt_rand(1000000,9999999).strstr($_FILES["address"]["name"],".");// 重 新 设置 升级 包 名 称 

S$address="./sjxz/".$path; /1 设置 升级 包 在 服务 器 中 存储 的 指定 路 径 

move_uploaded_file($_FILES["address"]["tmp_name"],$address); ”// 将 升级 包 上 传 到 指定 的 路 径 下 

S$address="./admin/sjxz/".$path; // 获 取 升 级 包 在 服务 器 中 的 存储 路 径 

include_once("../conn/conn.php"); /连接 数据 库 文件 


/将 上 传 的 数据 存储 到 数据 库 中 ， 这 里 将 升级 包 在 服务 器 中 的 路 径 存储 到 数据 库 中 
$query=mysqli_query($conn,"insert into tb_sjxz(name,typeid,content,addtime,address,bbid) values($name'， 
‘$typeid','$content','$addtime','$address','$bbid')"); 

if($queryX{ // 如 果 添加 操作 成 功 ， 则 弹出 提示 

echo "<script>alert(' 升 级 包 添加 成 功 ! ');history.back();</script>"; 


Jelse{ /如果 添加 操作 失败 ， 则 弹出 提示 
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echo "<script>alert( 升级 包 添加 失败 ! ');history.back();</script>"; 
} 


?> 





2.14.4 软件 升级 包 删 除 的 实现 过 程 


国 。 软件 升级 包 删除 使 用 的 数据 表 : tb_ bb、tb_ type、tb_sjxz 
软件 升级 包 删 除 的 实现 主要 根据 当前 数据 中 提供 的 ID， 执 行 delete 删除 语句 ， 将 数据 表 中 相同 ID 
的 数据 删除 。 其 运行 结果 如 图 2.45 所 示 。 





SNS oon: rset piet al 中 HHA 


图 2.45 软件 升级 包 删 除 的 运行 结果 


该 功能 主要 通过 editsjb.php 文件 和 deletesjb.php 文件 实现 。 通过 editsjb.php 文件 输出 数据 库 中 存储 
的 有 关 升 级 包 的 信息 ， 以 分 页 的 形式 显示 ， 在 每 条 记录 的 最 后 设置 一 个 删除 链接 ， 通 过 脚本 来 调用 
deletesjb.php 文件 ， 根 据 变量 中 的 ID 值 执行 删除 升级 包 的 操作 。 关 键 代 码 如 下 : 


倒 程 26 代码 位 置 ， 资源 包 \TM\02\becty365\admin\deletesjb.php 


<?php 

header("Content-type: text/html; charset=utf-8"); /设置 文件 编码 格式 
$id=$_GET[id]; /获取 变量 传递 的 ID 
include_once("../conn/conn.php"); /连接 数据 库 


// 执 行 删除 操作 ， 将 数据 表 中 对 应 的 ID 数据 删除 
if(mysqli_query($conn,"delete from tb_sjxz where id=".$id."™")){ 
echo "<script>alert(' 该 升级 包 删 除 成 功 !");history.back();</script>"，// 如 果 删 除 操作 成 功 ， 则 弹出 提示 
}else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 
echo "<script>alert(' 该 升级 包 删 除 失败 !");history.back();</script>"; 
} 


?> 


2.15 在 Linux 系统 下 发 布 网 站 


在 Linux 系统 下 发 布 基于 PHP 的 网 站 ,首先 需要 配置 PHP 的 运行 环境 ,其 次 需要 对 网 卡 参数 进行 
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设 定 。 这 里 将 以 发 布 “BCTY365 网 上 社区 ”网 站 为 例 讲解 在 Linux 系统 下 如 何 实现 网 站 的 发 布 。 假 设 
已 经 申请 到 表 2.13 所 示 的 网 络 参 数 。 


表 2.13 申请 到 的 网 络 参数 





























参数 值 
卫 192.168.1.* 
Netmask 2552552549 
Network 192.168.1.0 
Broadcast 192.168.1.* 
Gateway 192.168.1.* 
主机 名 Tsoft 
DNS 168.95.1* 


在 Linux 系统 下 网 站 发 布 的 操作 步骤 如 下 : 
(1) 配置 PHP 的 运行 环境 ， 在 2.4 节 中 已 经 做 了 详细 介绍 ， 这 里 不 再 装 述 。 
(2) 将 “BCTY365 网 上 社区 ”网 站 的 所 有 文件 复制 到 Apache 主 目录 下 。 
(3) 设置 主机 名 称 。 在 终端 窗口 中 输入 如 下 命令 编辑 /etc/sysconfig/network 文件 : 


Vi/etc/sysconfig/network 











将 该 文件 中 的 参数 NETWORKING 设置 为 yes, 表示 启动 网 络 , 将 参数 HOSTNAME 设置 为 Tsoft， 
表示 设置 主机 名 为 Tsoft。 
(4) 设置 网 卡 参数 。 在 终端 窗口 中 输入 如 下 命令 编辑 文件 /etc/sysconfig/network-scripts/ifcfg-eth0: 


Vi/etc/sysconfig/network-scripts/ifcfg-ethO 
该 文件 的 相关 参数 设置 如 表 2.14 所 示 。 
表 2.14 设置 网 卡 的 相关 参数 























参数 说 了 明 
DEVICE=eth0 设置 网 卡 名 称 ， 要 与 这 人 fe-eth0 对 应 
ONBOOT=yes 指定 在 开机 时 启动 网 卡 
BOOTPROTO=static 设 定 启动 时 获取 卫 的 方式 
IPADDR=192.168.1.* 设 定 服务 器 卫 地 址 
NETMASK=255.255.255.* 设 定子 网 掩 码 
BROADCAST=192.168.1.* 设 定 同 网 段 的 广播 地 址 
GATEWAY=192.168.1.* 设 定 网 卡 的 网 关 


(5) 设置 DNS 主机 的 卫 。 在 终端 编辑 /etc/resolvconf 文件 : 


Vi/etc/resolv.conf 





设置 参数 nameserver 的 值 为 168.95.1.*。 


Ti2 
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(6) 重新 启动 网 络 设置 。 在 终端 窗口 中 输入 如 下 命令 : 


letc/re.d/inin.d/network restart 
ifdow eth0 
ifup eth0 


(7) 打开 浏览 器 ， 在 地 址 栏 中 输入 服务 器 他 地 址 或 域名 ， 打 开 如 图 2.46 所 示 页 面 ， 则 说 明 在 Linux 
系统 下 “BCTY365 网 上 社区 ”网 站 发 布 成 功 。 
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EP | 











BCTY365 社 区 上 
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WE Tin - ° = 
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图 2.46 在 Linux 系统 下 “BCTY365 网 上 社区 ”网 站 运行 结果 


2.16 开发 技巧 与 难点 分 析 


2.16.1 ”管理 员 权限 的 设置 


为 了 更 好 地 管理 和 维护 网 站 的 论坛 ， 针 对 论坛 设置 了 一 个 管理 员 ， 该 管理 员 不 在 后 台 进 行 操作 ， 
而 是 在 前 台 为 管理 员 设置 特殊 的 权限 ， 也 可 以 称 之 为 版 主 。 其 实现 的 原理 是 : 首先 在 数据 库 中 设置 不 
同 的 值 代表 不 同 的 权限 ，0 代表 普通 会 员 ，1 代表 管理 员 ， 然后 在 论坛 的 页 面 中 进行 判断 ， 当 用 户 的 类 
型 为 1 时 ， 不 但 具有 普通 会 员 的 权限 ， 而 且 具 有 删除 发 布 帖子 、 回 复 帖子 和 项 帖 的 权限 ， 如 果 用 户 的 
类 型 不 是 1， 则 不 具有 上 述 的 权限 ， 只 能 是 发 布 和 回复 帖子 。 管 理 员 和 普通 会 员 登 录 的 页 面 效 果 是 不 同 
的 ， 如 图 2.47 和 图 2.48 所 示 。 








localhost 号 示 
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图 2.47 管理 员 登 录 的 操作 页 面 图 2.48 普通 会 员 登 录 的 操作 页 面 
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在 页 面 中 执行 的 判断 语句 判断 登录 用 户 的 类 型 ， 然 后 根据 类 型 判断 用 户 的 权限 。 程 序 关键 代码 如 下 : 


<?php 
// 判 断 当 前 用 户 是 否 具有 删除 帖子 的 权限 
if(isset($_SESSION["unc"])X{ 
/条 件 为 用 户 不 能 为 空 ， 并 且 是 管理 员 ， 才 具备 删除 帖子 的 权限 
$sqlu=mysqli_query($conn,"select usertype from tb_user where usernc=".$_SESSION["unc'"].""); 
$infou=mysqli_fetch_array($sqlu); 
if($infou["usertype"]==1X{ 1/ 判断 当 该 用 户 的 类 型 等 于 1 时 执行 下 面 的 内 容 
?> 
<!-- 调 用 JavaScript 脚本 ， 执 行 删除 发 布 帖子 的 操作 --> 
<img src="images/lt_15_10.jpg”onclick="javascript:if(window.confirm(' 您 确定 删除 该 帖 么 ?')==true}{window. 
location.href='bbs_delete.php?id=<?php echo $infob["id"]?>";}" style="cursor:hand"/> 
<?php 


} 
// 和 如果 用 户 的 类 型 不 是 1， 则 不 执行 上 述 的 内 容 


?> 





2.16.2 ”帖子 置顶 的 设置 


所 谓 帖子 置顶 就 是 将 某 个 指定 的 帖子 在 对 应 的 版 块 中 最 前 面 的 位 置 显示 ， 该 权限 只 有 管理 员 才 拥 
有 ， 普 通 会 员 不 具备 该 权限 。 其 实现 的 原理 如 下 : 

首先 ， 在 数据 库存 储 发 布 帖子 信息 的 数据 表 中 设置 一 个 字段 top， 指 定 该 字段 为 数字 类 型 ， 其 默认 
值 为 0。 

然后 ， 在 网 页 中 判断 登录 用 户 的 权限 ， 如 果 是 管理 员 ， 则 具有 帖子 置顶 的 权限 ， 否 则 将 弹出 提示 
对 话 框 “ 对 不 起 ， 您 不 具备 该 操作 权限 !”。 

最 后 ， 如 果 是 管理 员 ， 则 执行 settop.php 文件 ， 根 据 对 应 帖子 的 ID 查找 到 发 布 帖子 信息 表 中 对 应 
的 数据 ， 更 新 该 条 数据 中 top 字段 的 值 为 1。 

判断 登录 用 户 权 限 使 用 的 代码 如 下 : 


<?php 

// 如 果 $_SESSION["unc"] 的 值 为 空 ， 则 不 可 以 进行 项 帖子 的 操作 

if(lisset($_SESSION["unc"])X{ 
echo "javascript:alert(' 请 先 登 录 本 站 ,然后 进行 此 操作 ! ');window.location.href="index.php";"; 

Jelse{ 
// 否 则 将 判断 当前 用 户 的 类 型 ， 如 果 是 管理 员 则 可 以 顶 帖 
$sqlu=mysqli_query($conn,"select usertype from tb_user where usernc=".$_SESSION["unc"]."™™"); 
$infou=mysqli_fetch_array($sqlu); 


if($infou["usertype"]==1X{ /如果 用 户 的 类 型 为 1， 则 有 项 帖 的 权限 
echo "javascript:window.location.href="'settop.php?id=".$infob["id"]."™"; 
Jelse{ // 否 则 不 具备 该 权限 


echo "javascript:alert(' 对 不 起 ， 您 不 具备 该 操作 权限 ! ");"; 
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实现 帖子 置顶 是 通过 settop.php 文件 来 完成 的 , 在 该 文件 中 , 根据 变量 提交 的 值 获取 到 发 布 帖子 信 
息 表 中 对 应 的 数据 ， 更 新 该 条 数据 中 字段 top 的 值 ， 并 且 对 该 字段 的 值 进行 判断 。 如 果 字 段 top 的 值 为 
1， 则 说 明 该 帖 已 经 置顶 ， 此 时 将 字段 的 值 更 新 为 0， 即 取消 置顶， 如 果 字 段 top 的 值 为 0， 则 说 明 该 
帖 没 有 进行 置顶， 此 时 将 字段 的 值 更 新 为 1， 即 置顶 该 帖 。settop.php 文件 的 程序 代码 如 下 : 


<?php 
header("Content-type: text/html; charset=utf-8"); /设置 文件 编码 格式 
include_once("conn/conn.php"); /连接 数据 库 文件 


/根据 获取 的 ID 值 ， 从 数据 表 中 读 取 到 对 应 的 数据 
$sql=mysqli_query($conn,"select top from tb_bbs where id=".$_GET["id"]."™"); 
$info=mysqli_fetch_array($sql); 


if($info[top"]==1X{ // 判 断 对 应 数据 记录 中 字段 top 的 值 ， 如 果 字 段 top 的 值 为 1， 则 执行 下 面 的 内 容 
mysqli_query($conn,"update tb_bbs set top=0 where id=".$_GET["id].""); /更 新 字段 top 的 值 为 0 
jelseif($info[top]==0){ // 如 果 对 应 数据 记录 中 字段 top 的 值 为 0， 则 执行 下 面 的 内 容 


mysqli_query($conn,"update tb_bbs set top=1 where id=".$_GET["id].""); /更 新 字段 top 的 值 为 1 
} 
echo "<script>alert(' 置 上 设置 成 功 ! ');history.back();</script>"; 
?> 


2.17 在 线 支 付 技术 专题 


所 谓 在 线 支付 就 是 客户 端 (金融 机 构 需 客户 端 安装 由 金融 机 构 签发 的 数字 证 书 ， 信 用 卡 免 安 装 ) 
将 支付 信息 加 密 后 通过 互联 网 传送 到 支付 网 关 〈 支 付 网 关 是 解决 网 络 上 安全 支付 问题 的 交易 平台 ， 位 
于 互联 网 和 传统 的 金融 机 构 内 部 网 之 间 ， 其 主要 作用 是 将 互联 网 和 金融 网 络 安全 地 连接 起 来 ， 将 不 安 
全 的 网 上 交易 信息 传 给 安全 的 金融 网 络 ， 起 到 隔离 和 保护 金融 网 络 的 作用 )， 同 时 金融 机 构 网 上 支付 系 
统 反 馈 有 关 支 付 信息 ， 客 户 确认 无 误 后 进行 支付 确定 ， 支 付 网 关 负 责 商户 网 上 交易 资金 的 清算 ， 并 根 
据 商 户 提 供 的 开户 行 、 账 号 等 结账 信息 将 网 上 消费 款项 汇总 划 入 商户 账户 。 

BCTY365 网 上 社区 的 在 线 支付 是 与 中 国 工商 银行 合作 来 共同 完成 的 。BCTY365 网 上 社区 的 在 线 
支付 操作 步骤 如 下 : 

(1) 登录 网 上 社区 ， 如 图 2.49 所 示 。 














两 上 而 下 古山 





图 2.49 在 线 订 购 的 操作 页 面 
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(2) 购买 商品 。 在 本 页 中 ， 不 但 可 以 购买 商品 ， 还 可 以 查看 商品 的 详细 信息 和 购物 车 中 的 商品 信 
息 ， 如 图 2.50 所 示 。 

(3) 进入 购物 车 操作 页 面 。 在 该 页 面 中 ， 可 以 修改 购物 数量 、 删 除 指定 商品 、 清 空 购物 车 、 继 续 
购物 和 统计 购买 商品 的 金额 ， 也 可 以 单 击 “ 结 算 ” 按 钮 进入 到 商品 结算 页 面 ， 如 图 2.51 所 示 。 


名 _ 侣 ;_ 人 I 缠 程 央 典 
所 属 版 专 直 版 






网 w 价 。 将 : 2.000,00 元 











和 | 半价 元》 动 村 (个 ) 经 作 
版 机 : 吉林 省 明科 村 有 限 公司 ET | 2.00 om k | 
ED Gr Ess TI 主语 99 加 EL | 村 fy 时 | 刚 | 这 大 
于 民风 天 。 再 = 网 委 千 两 主导 计 : 5,999 oc 元 EE ] 
图 2.50 ”购买 商品 操作 页 面 图 2.51 购物 车 操作 页 面 


(4) 进入 到 购物 结算 页 面 ， 填 写 收 货 人 的 详细 信息 ， 确 认 后 提交 该 数据 ， 如 图 2.52 所 示 。 
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商品 世 征 单价 (元 》 未 时 4 机 ‘元 ) 
ot! A ee EE 人 
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图 2.52 填写 收 货 人 的 详细 信息 图 2.53 订单 确认 
(6) 进行 网 上 支付 。 在 这 里 可 以 选择 工行 网 上 支付 ， 也 可 以 选择 取消 该 订单 ， 如 图 2.54 所 示 。 
PT Enron 


图 2.54 执行 网 上 支付 


接 下 来 的 操作 在 工行 B2C 支付 页 面 上 进行 。 首 先 网 上 社区 按照 工商 银行 B2C 订单 数据 规范 形成 提 
交 数 据 ， 并 使 用 工商 银行 提供 的 API 和 商户 证 书 对 订单 数据 签名 ， 形 成 form 表单 返回 客户 浏览 器 ， 表 
单 action 地 址 指向 工商 银行 接收 商户 B2C 订单 信息 的 servlet; 然后 在 客户 确认 使 用 工行 网 上 支付 后 ， 
提交 此 表单 到 工商 银行 ， 最 后 工行 网 银 系 统 接收 此 笔 B2C 订单 ， 对 订单 信息 和 商户 信息 进行 检查 ， 通 
过 检查 则 显示 工行 B2C 支付 页 面 。 

客户 通过 工行 B2C 支付 页 面 实现 网 上 支付 ， 商 户 查询 网 上 银行 的 账户 ， 如 果 货 款 已 经 到 账 ， 则 根 
据 客户 指定 的 方式 将 货物 送 达 客户 手中 。 
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上 述 内 容 就 是 网 上 社区 系统 的 在 线 支付 流程 ， 涉 及 工商 银行 的 操作 内 容 这 里 不 做 讲解 。 这 里 主要 
讲解 如 何 将 订单 信息 提交 到 工商 银行 。 该 项 操作 主要 通过 shopping tiddphp 文件 来 实现 ， 首 先 从 数据 
库 中 读 取 订单 信 息 ， 然 后 将 订单 信息 进行 输出 ， 最 后 创建 “取消 订购 ”和 “工行 网 上 支付 ”两 个 超 链 
接 ， 通 过 JavaScript 脚本 来 调用 不 同 的 执行 文件 。 关 键 代码 如 下 : 


<?php include_once("conn/conn.php"); include_once("top.php"); /连接 数据 库 和 网 站 的 头 文件 ?> 
<!-- 省 略 了 部 分 代码 --> 

<?php 

@ $ddnumber=base64_decode($_GET["ddno"]); // 对 获取 的 订单 编号 进行 base64 解码 
// 获 取 该 订单 的 金额 信息 

$sql=mysqli_query($conn,"select * from tb_dd where ddnumber=".$ddnumber."™"); 
$info=mysqli_fetch_array($sql); 


$amount=$info["totalprice"]; 

@ $amount=str_replace(",",",number_format($amount,2)); /修改 数字 的 输出 格式 
$amount=str_replace(".",",number_format($amount,2)); /修改 数字 的 输出 格式 
?> 


“on // 省 略 部 分 HTML 代码 
<table width="630" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr><td width="159">&nbsp; 
<?php 
$sql=mysqli_query($conn,"select totalprice from tb_dd where ddnumber=".base64_decode($_GET 
["ddno"™]).™™); 
$info=mysqli_fetch_array($sql); 
echo "<font color=red><strong>".$info["totalprice"]."&nbsp; 元 </strong></font>"; 
?> </td> 
</tr> 
</table> 
i // 省 略 部 分 HTML 代码 
<script language="javascript"> 
// 打 印 订单 
© function openprintwindow(x,y,z){ 
window.open("printwindow.php?ddno="+x+"&pv="+z,"newframe","top=200,left=200,width=635,hei 
ght="+(230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no,status=no"); 


} 
</script> 
“oe /省 略 部 分 HTML 代码 
<table width="630" height="25" border="0" align="center cellpadding="0" cellspacing="0"> 
<tr> 
< 取消 该 订单 —> 





<td width="75"><img src="images/bg_14_14.jpg"” width="69" height="20" style="cursor:hand" onclick= 
"javascript:if(window.confirm(' 如 果 取 消 该 订单 ， 则 该 订单 将 被 删除 ， 您 需要 重新 购买 ! ')==true){window. location. 
href='deletedd.php?ddno=<?php echo $_GET["ddno"];?>";}"/></td> 

< 执行 网 上 支付 一 

<td width="125"><img src="images/bg_14_15.jpg" width="119" height="20" onclick="javascript:window. 
location.href='ddform.php?orderid=<?php echo base64_decode($_GETT"ddno"]);?>&amount=<?php echo $amount?> 
&orderDate=<?php echo date("Ymdhis");?>"," style="cursor:hand"/></td> 

</tr> 
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</table> 











< 上 省 略 了 部 分 代码 sa 
<?php include_once("bottom.php"); // 包 含 网 站 的 尾 文件 ?> 
< 代码 贴 十 
@ base64 decode0: PHP 实现 对 base64 编码 的 字符 进行 解码 。PHP 实现 字符 囊 的 base64 编码 通过 base64_encode() 


函数 。 
@ str replace(): 实现 字符 串 的 替换 。 该 函数 的 语法 如 下 : 


mixed str_replace(mixed search, mixed replace, mixed subject, int &count) 


str replace0 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 Teplace 替换 ， 参 数 &count 表示 替换 字符 串 执行 的 次 数 。 
@ openprintwindow0: JavaScript 脚 本 中 自 定义 的 函数 ， 用 于 执行 订单 的 打印 操作 。 


有 关 在 线 支付 流程 中 的 其 他 操作 实现 方式 已 经 在 2.10 节 中 进行 了 详细 的 讲解 ， 这 里 不 再 袭 述 。 其 
具体 的 代码 可 参考 本 书 的 资源 包 TMNVO2\bcty365\。 


2.18 本 章 总 结 


本 章 从 项 目 开 发 的 实际 角度 出 发 ， 以 某 科技 公司 的 实际 需求 为 背景 ， 详 细 地 讲解 “BCTY365 网 上 
社区 ”系统 的 开发 过 程 ， 其 中 以 系统 的 整体 开发 流程 为 主线 ， 重 点 介绍 技术 支持 、 在 线 订购 、 社 区 论 
坛 和 编程 词典 管理 等 几 个 大 模块 的 实现 方法 ， 并 且 对 管理 员 权 限 设置 、 帖 子 置 顶 设置 和 在 线 支付 技术 
进行 了 难点 分 析 和 专题 讲解 ， 而 且 在 本 章 中 还 讲解 了 在 Linux 系统 下 如 何 搭建 PHP 的 开发 环境 以 及 如 
何 发 布 网 站 。 


生生 = 
草 





办 公 自 动 化 管理 系统 


(Apachet+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


办 公 自 动 化 (Office Automation，OA) 系统 指 实现 办 公 室 内 事 
务 性 业务 的 自动 化 。 

办 公 自 动 化 没有 明确 的 定义 ， 最 普遍 的 说 法 是 凡是 在 传统 的 办 公 
室 中 采用 各 种 新 技术 、 新 机 器 、 新 设备 从 事 办 公 上 业务， 都 属于 办 公 自 
动 化 的 领域 。 

办 公 自 动 化 系统 与 办 公 自 动 化 在 概念 上 存在 一 定 的 差别 。 办 公 自 
动 化 通常 指 办 公 室 中 配备 具有 自动 化 功能 的 设备 ， 这 些 设备 能 使 某 些 
办 公 活 动 自动 化 或 实现 某 个 单位 业务 的 自动 化 处 理 ; 而 办 公 自 动 化 系 
统 则 是 在 办 公 自 动 化 功能 的 基础 上 发 展 起 来 ， 以 办 公 自 动 化 技术 为 主 
体 ， 同 人 、 组 织 、 制 度 、 环 境 等 相 结合 的 完整 的 系统 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

MH 办 公 自 动 化 管理 系统 的 开发 流程 

MW 进一步 学 习 如 何 做 项 目 需求 分 析 与 系统 设计 
”页 面 布局 中 的 框架 布局 

由 ”使 用 递归 图 数 做 多 级 下 拉 菜 单 配置 说 明 


Hi 系统 日 志 的 实现 
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3.1 开发 背景 














频 讲 解 


随 着 中 国 加 入 WTO 及 全 球 经 济 一 体 化 进程 的 加 快 , 世界 经 济 已 由 工业 化 经 济 逐 步 进入 网 络 信息 
化 时 代 。 在 信息 时 代 来 临 之 际 ， 各 企业 都 紧 跟 时 代 的 脚步 ， 转 变 着 企业 的 经 营 模式 、 管 理 模式 ， 从 
传统 的 人 工 管理 体制 向 信息 自动 化 管理 体制 过 渡 。 网 络 自动 化 办 公 系 统 就 是 在 这 样 的 大 背景 下 应 运 
而 生 的 。 

信息 时 代 的 到 来 让 人 们 尝 到 了 “信息 爆炸 ”的 滋味 ， 信 息 的 大 量 涌 入 让 企业 在 信息 处 理 方面 应 接 
不 暇 ， 传 统 的 办 公 模 式 对 信息 的 处 理 方法 早已 不 能 满足 企业 对 信息 快速 、 准 确 处 理 的 要 求 。 一 个 企业 
对 信息 数据 的 掌握 程度 、 处 理 能 力 ， 体 现 了 企业 对 市 场 的 敏感 程度 ， 数 据 的 真实 性 、 准 确 性 直接 决定 
着 企业 的 发 展 方向 。 从 传统 的 办 公 模 式 向 自动 化 办 公 管理 模式 转变 ， 提 高 企业 的 信息 处 理 能 力 ， 以 增 
强 企业 的 市 场 竞 争 力 ， 成 了 企业 发 展 过 程 中 的 首要 问题 。 








3.2 需求 分 析 


根据 与 客户 的 多 次 交谈 和 了 解 ， 本 系统 所 面向 的 客户 对 象 情况 如 下 : 

所 属 IT 行业 ， 目 前 主要 以 开发 门户 网 、 企 业 网 等 中 小 型 网 站 为 主 。 

公司 经 过 多 年 经 营 ， 已 经 相对 稳定 ， 并 有 了 自己 的 网 站 、 企 标 、 规 章 制度 和 基本 架构 。 

公司 规模 350 一 100 人 ， 人 手 一 机 ， 主 要 包括 部 门 经 理 、 普 通 职员 、 试 用 人 员 和 实习 人 员 。 

职员 所 用 计算 机 属于 局 域 网 内 网 ， 不 允许 访问 外 网 。 

公司 暂时 分 为 技术 部 、 人 事 部 和 质量 部 ， 但 不 排除 后 期 增加 其 他 部 门 的 可 能 。 

公司 实行 人 性 化 管理 ， 允 许 员 工 自由 发 表意 见 和 想法 ， 还 有 丰富 多 彩 的 活动 ， 如 比赛 、 旅 

游 等 。 

回 ”为 了 提高 工作 效率 ， 所 有 职员 定期 写 工 作 计划 ， 包 括 周 计 划 、 月 计划 ， 部 门 经 理 还 有 年 计划 
和 任务 计划 。 

回 ”根据 计划 的 完成 程度 和 完成 质量 ， 不 定期 地 选取 优秀 员工 。 


回回 罗 网 网 加 


3.3 系统 设计 


3.3.1 系统 目标 


本 系统 是 针对 中 小 型 企业 内 部 自动 化 办 公 管 理 的 要 求 进行 设计 的 ， 主 要 实现 如 下 目标 
回 ”键盘 操作 ， 快 速 响应 。 
回 ”实现 文件 类 信息 的 强大 的 管理 能 力 。 
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回 ”实现 对 员工 基础 信息 〈 人 事 消息 ) 的 管理 功能 等 。 

回 ”实现 个 人 办 公 的 信息 自动 化 管理 功能 。 

回 ”发 布 会 议 信息 ， 并 对 会 议 信息 进行 管理 

回 ”对 系统 用 户 进行 管理 

回 ”为 了 加 强 数据 保密 性 ， 为 每 个 用 户 组 设置 权限 级 别 。 

回 ”系统 最 大 限度 地 实现 了 易 安 装 性 、 易 维护 性 和 易 操 作 性 。 
回 ”系统 运行 稳定 、 安 全 可 靠 。 


3.3.2 ”系统 功能 结构 


根据 系统 分 析 ， 下 面 给 出 系统 的 前 、 后 台 功 能 结构 图 。 
办 公 自 动 化 前 台 功 能 结构 图 如 图 3.1 所 示 。 
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在 加 加 痢 度 

馈 划 | | 划 | | 划 划 下 
企业 公告 病 事假 
活动 安排 加 班 
消息 管理 考勤 记录 
















































































任务 绩效 ok \ 公司 简介 
质量 绩效 DD CD 规章 制度 

































































优秀 员工 组 织 结构 
绩效 评定 职 个 | | 账 | | 权 企业 管理 
员 | | 间 | | 人 | “| 号 | | 限 
资 | | 箱 | | 设 | “| 权 | | 分 
料 定 | “| 限 | | 配 



































图 3.1 办 公 自 动 化 前 台 管 理 系统 
办 公 自 动 化 后 台 功能 结构 图 如 图 3.2 所 示 。 
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3.2 办 公 自 动 化 后 台 管 理 系统 
3.3.3 ”系统 功能 预览 


办 公 自 动 化 管理 系统 由 多 个 功能 模块 组 成 ， 下 面 仅 列 出 几 个 典型 功能 的 页 面 ， 其 他 页 面 参见 资源 


包 中 的 源 程序 。 


前 台 登 录 界 面 如 图 3.3 所 示 , 该 页 面 用 于 实现 对 用 户 登 录 的 用 户 名 和 密码 进行 验证 。 企 业 信息 页 面 


如 图 3.4 所 示 ， 该 页 面 用 于 显示 企业 文化 和 各 规章 制度 。 


数据 备份 

















修改 密码 























3.3 ”前 台 登 录 〈 资 源 包 \TM\03\oa\index.php) 3.4 ”企业 信息 (资源 包 \TM\03\oa\qyxx\r_system.php) 


@_ 








第 3 章 ”办公 自 动 化 管理 系统 ( Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


个 人 计划 页 面 如 图 3.5 所 示 , 该 页 面 主要 实现 用 户 的 工作 反馈 和 周 、 月 计划 等 功能 。 后 台 登 录 界面 
如 图 3.6 所 示 ， 该 页 面 用 于 管理 员 后 台 登 录 。 
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图 3.5 个 人 计划 (资源 包 \TM\03\oa\grjh\person plan.php) ”图 3.6 后 台 登 录 (资源 包 \TM\03\oa\admin\index.php) 


职员 管理 页 面 如 图 3.7 所 示 ， 该 页 面 主要 用 于 实现 对 职员 的 查询 与 修改 等 功能 。 权 限 分 配 页 面 如 
图 3.8 所 示 ， 该 页 面 用 于 对 功能 使 用 权 的 分 配 。 


登录 | 企业 公告 | 帮助 | 进出 
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一 -一 一 ja 
由 号 权 阳 = 
本 [7 np fp 
| PF Tip 性 改 /投降 
| 
下 
和风 
图 3.7 职员 管理 图 3.8 权限 分 配 
(资源 包 \TM\03\oa\admin\zyel\show_staf.php) (资源 包 \TM\03\oa\admin\qxegl\pur_assign.php) 


3.3.4 ”系统 流程 图 


用 户 在 登录 一 个 系统 后 会 进行 一 系列 的 操作 ， 把 这 些 操作 的 过 程 和 结果 以 图 形 的 形式 表现 出 来 ， 
这 就 是 系统 流程 图 。 一 个 好 的 流程 图 ， 不 仅 可 以 让 开发 者 迅速 地 理 清 思路 、 及 时 解决 出 现 的 问题 ， 也 
可 以 让 使 用 者 很 快 明白 该 系统 的 操作 方式 与 方法 。 办 公 自 动 化 管理 系统 的 工作 流程 图 如 图 3.9 所 示 。 
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图 3.9 ”办公 自动 化 管理 系统 流程 图 
3.3.5 ”开发 环境 
在 开发 办 公 自动 化 管理 系统 平台 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 





















加 ”操作 系统 :Windows 7 及 以 上 /Linux。 攻 | 
加 ”集成 开发 环境 : phpStudy。 Toe a 
回 PHP 版 本 : PHP7.0。 和 . 后 台数 据 库 锋 接 广 件 
回 数据库 ， MySQL 5.5。 外 和 
开发 工具 : PhpStorm。 26 后 js 本文 
浏览 器 :谷歌 浏览 器 。 Pe: ti 
[= 村: 职员 管理 模块 
conn 前 台数 据 库 链接 文件 
3.3.6 ”文件 夹 组 织 结构 和. tr 
学 2 和 和 
系统 功能 结构 和 系统 流程 设计 完成 后 ， 接 下 来 Een 
设计 网 站 的 文件 夹 结构 ， 合 理 的 文件 夹 结构 不 仅 易 二 信和 本 
于 快速 地 开发 ， 对 后 期 系统 的 调试 、 维 护 和 管理 也 de 
能 起 到 事半功倍 的 效果 。 本 系统 的 文件 夹 组 织 结构 一 
如 图 3.10 所 示 。 图 3.10 办 公 自 动 化 管理 系统 文件 夹 组 织 结构 


& 
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3.4 数据 库 设 计 











基于 目前 的 系统 开发 ， 没 有 数据 库 的 支持 根本 是 无 法 想象 的 ， 办 公 自 动 化 管理 系 
统 更 是 使 用 了 大 量 的 数据 表 来 存储 数据 。 本 节 就 对 办 公 自 动 化 管理 系统 所 需要 使 用 的 数据 库 进行 说 明 
与 设计 。 
3.4.1 数据 库 分 析 

由 于 本 系统 采用 的 是 PHP 语言 ， 数 据 库 理所当然 地 使 用 MySQL， 不 仅 是 因为 开发 成 本 低 ， 更 重 


要 的 是 两 者 之 间 的 默契 程度 和 稳定 程度 要 远 远 高 于 其 他 的 数据 库 组 合 ， 并 且 MySQL 数据 库 对 于 一 个 企 
业 的 内 部 办 公 自 动 化 管理 系统 完全 够 用 。 


3.4.2 ”数据 库 概念 设计 


根据 以 上 的 需求 分 析 及 系统 分 析 ， 规 划 出 办 公 自 动 化 管理 系统 主要 的 几 个 实体 关系 E-R 图 。 

1. 用 户 信息 实体 

用 户 信 息 实体 包括 职员 的 账号 、 密 码 、 姓 名 、 电 话 等 表示 个 人 身份 的 数据 资料 ， 如 图 3.11 所 示 。 
2. 考勤 登记 实体 


考勤 登记 实体 包括 登记 日 期 、 登 记 时 间 、 登 记 类 别 〈 上 班 或 下 班 )、 登 记 状态 〈 人 迟到 或 早退 ) 和 登 
记 人 、 备 注 等 信息 ， 如 图 3.12 所 示 。 























CEaD 
eH 
考勤 登记 
备注 登记 状态 
登记 人 
图 3.11 用 户 信息 实体 E-R 图 图 3.12 考勤 登记 实体 E-R 图 


3.4.3 ”数据 库 物 理 结构 设 计 
在 本 系统 中 创建 了 一 个 数据 库 db_office， 一 共 包 含 13 个 数据 表 ， 如 图 3.13 所 示 。 
限于 篇 幅 ， 这 里 只 给 出 主要 的 表 结 构 和 表 说 明 ， 其 他 数据 表 结 构 请 参见 本 书 附带 的 资源 包 。 
1. tb_users〈 用 户 列表 ) 


用 户 列表 主要 用 于 存储 职员 的 姓名 、 人 性别 等 私人 信息 及 部 门 、 工 作 组 等 与 公司 相关 的 公共 信息 。 
该 数据 表 结构 如 图 3.14 所 示 。 
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四 服务 器 : localhost 》 晶 数据 库 : db_office 
表 类 整理 党 明 
也 _company MyISAM gb2312_chinese_ci 企业 信息 列表 


图 服务 器 : localhost > 局 数据 库 : db_office ， 司 表 :tb_users 
也 _controller MyISAM gb2312_chinese_ci 管理 员 列 表 





了 及 天 整理 性 性 Nul 树 认 要 外 说 明 
了 depart 。 MySAM gb2312_chinese_ci 部门 首 更 列 天 pr pr 3 pe rr 
onp | MON | ob2S12.chinesa_ di | 用户 拓 管理 和 user 。 Varrhar(50) gb2312_chinese ol 否 月 F 哑 号 
tb ss MyISAM 。 gb2312_chinese_cl 。 审核 列表 Fe 否 1 
了 bst MiSAM gb2312_chinese_ci 功能 列表 uname varhar(20) gb2312_chinese 是 NULL 用 户 姓名 
i MYISAM gb2312_chinese_ci 意见 条 刘表 eee sharf) 。 92312_ehinese ol 是 NL 用 户 性 别 
也 _person 。 WWISAM gb2312_chinese_cl 人事 列表 a 是 ”NULL 用 户 生日 
th_plan MyISAM gb2312_chinese_ci 。 计 和 衣 .adiress varhert50) 902312_chinese_cl 是 ”NULL 用 户 住址 
也 _register 。 MyISAM gb2312_chinese_ci 时 记 列表 utel varhar(2) gv2312_chinese_ci 是 NULL 用 户 电 话 
tb_setup MyISAM gb2312_chinese_ci 时 间 列表 omal 。 rarhart50) gv2312 chinese_cl 是 NULL 用 户 mail 
ern MADAM [ob2312.ehinesazel | 优 汪 员工 用 udepart 。 yarrhar(20) gb2312_chinese_ci 否 斯 属 部 门 
tb users 。。 MISAM gb2312_chinese_ci 用 户 天 ison nt) 再 种 洒 几 号 
图 3.13 ”db_office 数据 库 中 的 数据 表 列表 图 3.14 用 户 列表 


2. tb_person (人 事 列表 ) 


人 事 列表 用 于 存储 人 事 部 门 发 布 的 信息 ， 如 信息 标题 、 信 息 内 容 、 信 息 时 间 等 。 该 数据 表 结构 如 
3.15 所 示 。 


胃 服务 器 :localhost > 加 数据 库 : db_offlce ，》 回 表 :th_parson 


有 类型 于 民团 Nul 器 认 要 外 襄 明 
md init) 至 uio_nprement 上 动 和 
pe varsharG0) gb2312_chinese_el 本 么 首要 
Pp-contont medurrtog gb2312_ cninose_ el 再 人 性 内 容 
pime ob 百 MH 
re 机 LT 


图 3.15 人 事 列表 


3.5 ”公共 模块 设计 








视频 讲解 


本 系统 的 公共 模块 包括 conn 数据 库 链接 文件 、ess 样式 文件 、JavaScript 脚本 文件 以 及 inc 下 的 自 
定义 函数 文件 和 包含 文件 等 几 类 文件 , 其 中 数据 库 链接 文件 和 css 样式 文件 在 前 面 的 章节 中 有 过 系统 的 
介绍 ， 这 里 主要 讲解 本 系统 所 涉及 的 JavaScript 脚本 文件 和 部 分 自 定义 函数 文件 。 


3.5.1 JavaScript 脚本 


在 办 公 自 动 化 管理 系统 中 ，JavaScript 脚本 一 般 用 于 表单 元 素 验证 ， 如 判断 text 文本 框 输入 是 否 为 
空 ， 输 入 格式 是 否 符合 标准 等 。 在 网 页 中 使 用 JavaScript 脚本 的 方式 主要 有 以 下 3 种 : 
(1) 在 网 页 中 使 用 <script></scrip 人 标签 对 。 
<script></scrip 亿 标签 对 可 以 放 在 网 页 的 任意 位 置 ， 一 般 放 在 <head></head> 或 <body></body> 之 间 。 
代码 如 下 : 
<head> 


<script>document.write(" 办 公 自动 化 管理 系统 开发 ");</script> 
</head> 


全 
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(2) 在 单独 文件 中 使 用 。 

如 果 JavaScript 脚本 比较 多 ， 而 且 位 置 分 散 不 易 管 理 ， 可 以 统一 放 到 一 个 扩展 名 为 js 的 文件 中 ， 
使 该 文件 成 为 JavaScript 脚本 文件 。 在 脚本 文件 中 ,不 需要 使 用 <script></script> 标 签 对 ， 直 接 写 脚本 代 
码 即 可 。 当 有 页 面 需 要 使 用 到 里 面 的 JavaScript 脚本 时 ， 可 以 这 样 引用 ， 代 码 如 下 ; 


<script src="ad _js.js" jangauge="javascript"></script> 


(3) 在 表单 元 素 或 标签 中 使 用 。 
这 是 最 直接 的 使 用 方式 ， 如 果 是 少量 的 脚本 则 可 以 这 样 使 用 。 例如， 在 超 链 接 标签 <a> 中 想 要 使 用 
JavaScript 脚本 ， 代 码 格式 如 下 : 


<a href="#" onclick="alert('hello');">hello</a> 


在 办 公 自 动 化 管理 系统 中 ， 这 3 种 方法 都 有 使 用 到 ， 在 后 面 涉及 具体 应 用 时 再 进行 说 明 。 下 面 先 
来 看 两 个 经 常 使 用 到 的 脚本 函数 。 

(1) 验证 函数 ， 用 于 判断 表单 元 素 是 否 为 空 。 如 果 为 空 ， 则 返回 false， 并 将 光标 焦点 定位 到 出 问 
题 的 表单 元 素 。 程 序 代码 如 下 : 


倒 程 01 代码 位 置 : 资源 包 \TM\03\oa\js\client js.js 








/后 台 登 录 界 面 验证 脚本 
function check(){ 
if(login.username.value=="){ 1/ 判断 用 户 名 是 否 为 空 
alert(" 请 输入 用 户 名 !"); 
login.username .focus(); /将 光标 焦点 定位 到 该 表单 元 素 
return false; 
} 
if(login.pwd.value=="™")}{ 1/ 判 断 用 户 密码 是 否 为 空 
alert(" 请 输入 密码 册 ); // 如 果 为 空 ， 弹 出 提示 框 
login.pwd.focus(); /将 光标 焦点 定位 到 该 表单 元 素 
return false; /返回 false 


} 
} 


(2) 选择 框 函数 ， 用 于 删除 确认 。 当 要 执行 删除 操作 时 ， 使 用 JavaScript 脚本 文件 进行 确认 ， 以 
免 因为 误 操 作 而 引起 不 必要 的 损失 。 


倒 程 02 ”代码 位 置 : 资源 包 \TMMVO3\oaNjsvclient_js.js 


function cfm(){ 
说 confirm(' 确 认 要 删除 吗 ? )) // 选 择 框 函数 
return true; // 如 果 选 择 确认 ， 则 返回 true， 继 续 执行 
else 
return false; // 否 则 ， 返 回 false 


有 





3.5.2 自 定 义 函数 
在 inc 文件 夹 下 ， 有 两 个 主要 文件 : chec php 文件 和 func.php 文件 。 其 中 ，chec.php 文件 是 权限 检 
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查 文件 ， 是 办 公 自 动 化 管理 系统 每 个 页 面 都 要 引用 的 ，func.php 文件 是 自 定义 函数 文件 。 
1. chec.php 文件 


对 办 公 自 动 化 管理 系统 来 说 ， 用 户 对 每 个 页 面 的 访问 都 要 有 该 页 的 权限 才 可 以 ， 如 果 没 有 权限 级 
别 的 限制 ， 随 意 访问 重要 的 资源 和 数据 ， 那 么 只 能 说 明 这 是 个 完全 失败 的 系统 。chec.php 文件 的 代码 
如 下 : 


倒 程 03 ”代码 位 置 : 资源 包 \TM\03\oa\inc\chec.php 
<?php 
session_start(); /开启 session 支持 
@ if(lisset($_SESSION[u_name')) 
echo "<script>alert(' 您 无 权 访问 ');location="../index.php';</script>"; 
@ if($_SERVER['HTTP_REFERER'] =="") 
echo "<script>alert(' 本 系统 不 允许 从 地 址 栏 访问 ');history.go(-1);</script>"; 
?> 





< 代码 贴 十 

@ isset0 函 数 : 用 来 判断 session["u_name'] 是 否 被 创建 ， 如 果 Session 已 经 创建 ， 返 回 true， 否 者 返回 false。 其 中 
session[_name'] 是 用 户 登 录 验 证 通过 后 创建 的 ， 如 果 没 有 登录 的 步骤 或 Session 超时 ， 那 么 就 会 提醒 用 户 没有 权限 。 

@ $_SERVER['HTTP_REFERER']: 系统 预定 义 变量 ， 存 储 的 是 上 一 页 的 URL 地 址 。 这 个 判断 的 作用 是 防止 当前 用 
户 访问 不 属于 自己 权限 内 的 网 页 . 


2. func.php 文件 

限于 篇 幅 ， 这 里 只 给 出 两 个 自 定义 函数 的 代码 和 说 明 ， 其 他 函数 请 到 资源 包 中 自行 查看 。 
(1) 读 取 字 段 函数 read_field0。 

该 函数 的 作用 是 根据 传 入 的 id 值 取得 相关 的 字段 名 ， 一 共有 4 个 参数 ， 函 数 代码 如 下 : 


倒 程 04 代码 位 置 ， 资源 包 \TM\03\oa\inc\func.php 


function read_field($conn, $tablename, $fieldname, $n_id}{ 
S$sqlstr = "select ".$fieldname." from ".$tablename." where id = ".$n_id; /生成 SQL 语句 





Sresult = mysqli_query($conn ,$sqlstr); /执行 SQL 语句 
Srows = mysqli_fetch_row($result); // 返 回 结果 集 
return $rows[0]; // 返 回 需要 的 字段 名 


} 
read_field0 函 数 的 参数 说 明 如 表 3.1 所 示 。 
表 3.1 read_field() 函 数 的 参数 说 明 








参数 名 称 参数 说 了 明 
Sconn 数据 库 链 接 资 源 变 量 
Sfieldname 要 查找 的 字段 名 
S$tablename 要 查找 的 表 名 

Sn id 要 查找 的 这 值 
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(2) 处 理 消 息 函 数 re_ message0 。 
该 函数 的 作用 是 返回 消息 结果 ， 一 共有 2 个 参数 ， 函 数 代码 和 参数 解释 如 下 : 


倒 程 05 代码 位 置 ， 资源 包 \TM\03\oa\admin\inc\func.php 


function re_message(S$result,$l_address)}{ 
if($result) 
echo "<script>alert(' 操 作成 功 ! ');location=".$I_address.";</script>"; // 如 果 结果 为 true, 转 到 其 他 页 面 
else 
echo "<script>alert(' 系 统 繁忙 ， 请 稍 后 再 试 ');history.go(-1);</script>"; // 如 果 结 果 为 false， 回 到 上 一 步 





加 ”$result: 数据 库 返 回 结果 , 如 果 返 回 值 为 true, 说 明 对 数据 库 的 操作 成 功 ; 如 果 返 回 值 为 false， 
说 明 操作 失败 。 
回 ”$l address: 操作 成 功 后 ， 要 返回 的 URL 地 址 。 


3.6 前 台 首 页 设计 








;视频 讲解 
对 于 一 个 办 公 自 动 化 管理 系统 来 说 ， 首 页 不 可 能 ， 也 不 允许 显示 太 多 的 内 容 和 复 


杂 的 设计 ， 因 为 办 公 自动 化 管理 系统 的 作用 是 帮助 提高 企业 的 管理 效率 和 职员 的 工作 效率 。 如 果 页 面 
过 于 烦琐 ， 会 让 职员 分 不 清 主 次 ， 找 不 到 重点 。 所 以 一 定 要 突出 重点 内 容 ， 显 示 关键 功能 。 


3.6.1 前台 首页 概述 


根据 用 户 对 各 个 功能 模块 的 使 用 频率 和 重要 程度 ， 本 系统 的 首页 中 要 显示 的 模块 主要 有 以 下 3 
部 分 。 

回 ”网 站 首部 导航 栏 : 包括 当前 用 户 、 所 在 部 门 、 当 前 时 间 、 首 页 、 重 新 登录 和 退出 登录 。 

回 ”网 站 左 侧 导 航 栏 : 包括 各 个 管理 模块 及 分 类 。 

> “企业 信息 模块 : 包括 公司 简介 、 规 章 制度 、 组 织 结构 和 企业 管理 。 

企业 绩效 模块 : 包括 任务 绩效 、 质 量 绩效 和 绩效 评定 。 
人 事 消息 模块 : 包括 企业 公告 、 活 动 安排 和 消息 管理 。 
审核 批示 模块 : 包括 发 布 审核 和 批示 审核 。 
考勤 管理 模块 : 包括 上 下 班 登记 、 病 事假 登记 、 加 班 登记 和 考勤 设置 。 
个 人 计划 模块 : 包括 工作 反馈 、 周 计划 、 月 计划 、 年 计划 和 任务 计划 。 

> ”职员 天 地 模块 : 包括 职员 浏览 、 意 见 箱 和 个 人 设 定 。 
回 ”网 站 主 显示 区 : 默认 显示 企业 公告 、 活 动 安排 、 个 人 计划 和 审核 批示 。 
本 案例 中 提供 的 前 台 首页 如 图 3.16 所 示 。 该 首页 在 本 书 资源 包 中 的 路 径 为 TM\03\oapub_mainphp。 


VY VV YY Y 
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3.16 ”办 公 自 动 化 管理 系统 前 台 首 页 


3.6.2 ”前 台 首 页 技术 分 析 


本 系统 前 台 页 面 的 布局 使 用 的 是 Flex 布局 。 使 用 Flex 布局 可 以 简便 、 完 整 、 响 应 式 地 实现 各 种 页 


面 布局 。 采 用 Flex 布局 的 元 素 ， 称 为 Flex 容器 (flex container)， 简 称 “ 容 器 ”。 它 的 所 有 子 元 素 
成 为 容器 成 员 ， 称 为 Flex 项 目 (flex item)， 简 称 “ 项 目 ”。 


1. Flex 布局 格式 
框架 布局 的 格式 很 简单 ， 只 要 几 行 代码 即 可 ， 常 用 的 格式 如 下 : 


自动 





<?php include('top.php') ?> 
<div class="container"> 
<?php include('left.php') ?> 
<div class="right"> 
<!-- 省 略 其 余 代 码 --> 
</div> 
</body> 
</html> 


其 中 top.php 是 顶部 内 容 ，leftphp 是 左 侧 菜单 。Class='Container' 是 容器 ， 而 class=left 和 class='right" 


是 容器 成 员 。 
2. Flex 属性 


在 前 台布 局 页 面 的 container 容器 中 ， 左 侧 菜 单 大 小 保持 不 变 ， 而 右 侧 可 以 自由 伸缩 。CSS 布 
式 如 下 : 
.Container { 


display: flex; 
display: -webkit-flex; 


全 





局 样 
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flex-flow: row wrap; 
-ms-flex-flow: row wrap; 
-webkit-flex-flow: row wrap; 


} 


left { 
margin: Opx; 
width: 203px; 
height: 505px; 
text-align: center; 
background:url(../images/left.jpg) 
} 


right { 
flex: 1; 
-webkit-flex: 1; 
-ms-flex: 1; 
margin:0 20px ; 





3.6.3 前台 首 页 的 实现 过 程 


国 。 前台 首页 使 用 的 数据 表 : tb users、tb list、tb person、tb_register、tb_plan 

办 公 自 动 化 管理 系统 的 前 台 首 页 采用 的 是 二 分 栏 结构 布局 。 二 分 栏 布局 的 特点 是 简洁 、 大 气 、 个 
性 鲜明 ， 其 框架 设计 依据 其 内 容 形式 的 变化 而 灵活 多 变 ， 结 构 简练 ， 非 常 符 合 办 公 自 动 化 管理 系统 的 
风格 。 前 台 首 页 关键 代码 如 下 : 


倒 程 06 ”代码 位 置 : 资源 包 \TM\03\oa\pub_main.php 
<?php 
© include "inc/chec.php"; 
© include "conn/conn.php"; 
?> 
<?php include(top.php') ?> 
<div class="container"> 
<?php include('left.php') ?> 
<div class="right"> 
<div class="title"> 
最 新 消息 
</div> 
/省 略 其 余 代码 
</body> 
</html> 


< 人 代码 贴 十 


@ inc/checphp: 引入 登录 检测 函数 。 
@ conn/conn .php: 引入 数据 库 连 接 函 数 。 
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3.7 人 事 消 息 模 块 设计 














视频 讲 





人 事 消息 模块 主要 是 对 文件 进行 收发 管理 ， 模 块 的 设计 和 实现 并 不 复杂 ， 但 在 整个 系统 中 的 位 置 
却 很 重要 。 因 为 公告 栏 和 意见 箱 面向 的 是 全 体 用 户 ， 企 业 的 最 新 动向 和 职员 的 意见 想法 首先 都 是 在 这 
里 体现 出 来 的 。 


3.7.1 ”人事 消息 模块 概述 


人 事 消息 模块 主要 包含 了 两 部 分 内 容 : 一 部 分 是 面向 全 体 用 户 的 ， 包 括 查 看 公告 、 活 动 、 意 见 箱 
等 ， 另 一 部 分 是 仅 对 人 事 部 开放 ， 其 他 用 户 不 允许 、 也 不 会 看 到 的 页 面 ， 如 公告 管理 、 意 见 管理 等 。 
人 事 消息 用 例 图 如 图 3.17 所 示 。 





发 布 公告 一 一 | 
超级 用 户 [> ! 普通 用 户 


图 3.17 人 事 消息 用 例 图 
3.7.2 人事 消息 模块 技术 分 析 


在 人 事 消息 模块 中 ,企业 公告 和 活动 安排 都 是 在 同一 显示 页 中 显示 〈(p_message.php), 为 了 区 分 不 
同 的 操作 ， 需 要 给 显示 页 (p_message.php) 传 一 个 u_id 值 ， 在 数据 表 中 根据 不 同 的 u_id 值 ， 取 得 不 同 
类 别 的 内 容 。 传 值 方 式 有 url 传 值 方式 (如 p_message.php?u id=1 ) 和 表单 传 值 方式 。 

同样 地 ， 在 接收 页 中 ， 接 收 方式 也 有 两 种 ，get 方式 和 post 方式 。PHP 中 使 用 预定 义 变量 $_GET 
和 $_POST 来 接收 传 值 ， 格 式 如 下 : 


$_GETIPOST[u_id]; 1/ 注意， 变量 名 都 为 大 写 


对 于 ul 传 值 方式 ， 接 收 页 始终 用 $_GET[] 变 量 来 接收 ， 如 果 使 用 form 表单 传 值 ， 那 么 就 要 看 form 表 
单 中 method 属性 的 设置 了 。 如 果 method=get， 则 使 用 $_GET[] 变 量 ， 如 果 method=post， 则 使 用 $_POST[] 


他 
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变量 。 
PHP 对 传 值 方式 和 接收 方式 规定 很 严格 ， 用 $_POST[m id] 接收 不 了 p_message.php?u id=1 传 过 来 
的 参数 。 同 样 地 ， 用 $_GET[u_ id] 也 接收 不 了 method=post 的 表单 元 素 值 。 


$s 注意 如 果 在 php.ini 配置 文件 中 GLOBAL=ON/OFF 这 行 的 值 为 ON, 那么 直接 写 $name 就 可 以 
调用 表单 元 素 的 值 ， 而 不 区 分 get 和 post; 如 果 为 OFF， 则 不 可 以 。 直 接应 用 表单 名 称 十 分 方便 ， 
但 也 存在 着 安全 隐患 。 推 荐 关闭 GLOBAL 。 


3.7.3 消息 管理 的 实现 过 程 


国 。 消息 管理 模块 使 用 的 数据 表 : tb_person 

单 击 “ 人 事 消 息 ” 选 项 中 的 “消息 管理 ” 子 选项 ， 在 主 框架 (mainFrame) 内 就 会 显示 消息 管理 页 
面 ， 在 页 面 中 显示 发 布 过 的 消息 列表 和 对 消息 的 操作 ， 包 括 “修改 “删除 ”和 “发 布 新 消息 ”。 消 息 
管理 页 面 的 运行 结果 如 图 3.18 所 示 。 





稍 息 管理 | 
Eo] Ba ES 
2007-1204 it /We 
207-1204 Duinarsg S/W 
20171221 FE W/Wie 


ee 
EA 
Fr 
Ea 

es 


3.18 ”消息 管理 页 面 运 行 结果 


在 本 页 面 中 ， 系 统 首先 通过 权限 检查 文件 (chec.php》 判 断 用 户 是 否 为 登录 用 户 ， 是 否 为 非法 链接 。 
如 果 被 系统 判断 为 非法 操作 ， 系 统 将 直接 关闭 ， 如 果 没 有 异常 情况 ， 则 系统 继续 执行 下 面 的 代码 (在 后 
面 的 例 程 中 ， 几 乎 每 个 页 面 都 有 chec.php 文件 ， 如 果 不 是 特殊 说 明 ， 则 都 为 正常 登录 ， 到 时 将 不 再 具体 
说 明 )， 当 用 户 通 过 检查 后 ， 系 统 将 访问 数据 库 ， 从 人 事 消 息 数据 表 中 tb_person) 读 取出 消息 发 布 时 间 、 
消息 标题 ， 显 示 在 页 面 中 ， 并 通过 id 为 每 条 数据 设置 “修改 ”和 “删除 ”操作 。 程 序 关键 代码 如 下 : 


倒 程 07 代码 位 置 ， 资源 包 \TM\03\oarsxx\p_manage.php 
<?php include("../top.php') ?> 
<div class="container"> 
<?php include('../left.php') ?> 
<div class="right"> 
<table width="765" border="0" cellpadding="0" cellspacing="0" class="big_td"> 


$ 


<td height="33" background="../images/list.jpg" id="list"> 消 息 管理 
</td> 
< 
</table> 


3 
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<table width="765" border="0" cellspacing="0" cellpadding="0" bgcolor="#DEEBEF" class="big_td"> 
<tr> 
<td width="100" height="25" align="center" valign="middle" scope="col"> 发 布 时 间 </td> 
<td height="25" align="center" valign="middle" scope="col"> 标 题 </td> 
<td width="150" height="25" align="center" valign="middle" scope="col"> 操 作 </td> 
</tr> 
<?php 
S$sqlstr = "select id,p_time,p_title from tb_person"; 
Sresult = mysqli_query($conn,$sqlstr); 
while($rows = mysqli_fetch_row($result)}{ 
echo "<tr>"; 
for($i=1;$i<count($rows);$i++X{ 
echo "<td height=30 style='text-indent: 30px;'’>". $rows[$i]."</td>"; 
} 
echo "<td><a href=',m_message.php?id=".$rows[0]."> 修改 </a>/<a href='d_message_chk. 
php?id=".$rows[0]." onclick='return del_mess();> 删 除 </a></td>"; 
echo "</tr>"; 
} 
和 
<tr> 
<td height="30" align="right" valign="middle" colspan="3"> 
<a href='add_manage.php' target="mainFrame"> 发 布 新 消息 </a> 
</td> 
</tr> 
</table> 
</div> 





下 面 分 别 介 绍 “ 发 布 新 消息 ””“ 修 改 ” 和 “删除 ”消息 的 实现 过 程 。 

1. 发 布 新 消息 

单 击 “发 布 新 消息 ” 超 链接 ， 进 入 发 布 页 面 。 发 布 消 息 页 面 的 运行 结果 如 图 3.19 所 示 。 
OA 办 公 自 动 化 管理 系统 


Me 

















EECETTE3S3 














图 3.19 ”发 布 消息 页 面 运行 结果 
发 布 页 面 中 ， 主 要 包含 一 个 消息 表单 元 素 ， 部 分 表单 元 素 如 表 3.2 所 示 。 


全 
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表 3.2 发 布 消息 页 面 的 主要 表单 元 素 










发 布 消息 表单 
消息 标题 
消息 内 容 


消息 类 型 









action="add manage chkphp" method="post" id="addmess" 
name="p title" id="p title" 








p_content name="p_content" id=-"p content" cols="60" rows="15" 
<option value="9"> 企 业 公 告 </option> 
<option value="10"> 活 动 安排 </option> 


value=" 发 布 " onclick="return add_mess():" 





P_type 








submit2 





在 消息 页 面 中 输入 消息 的 相关 内 容 后 , 单 击 “ 发 布 ”按钮 , 系统 将 会 跳 到 消息 处 理 页 (add_manage_ 
chk.php) 进行 处 理 ， 将 消息 存储 到 数据 库 中 ， 然 后 再 返回 该 页 。 程 序 关键 代码 如 下 : 


倒 程 08 ”代码 位 置 ， 资源 包 \TM\03\oairsxx\add_manage_chk.php 


<?php 

session_start(); 

include "../inc/chec.php"; 

include "../conn/conn.php"; 

include "../inc/func.php"; 

S$sqlstr = "insert into tb_person values(",".$_POST[p_title].",".$_POST[p_content].",now(),".$_POST[p_ 
type].™)", 

Sresult = mysqli_query($conn,$sqlstr); 

re_message($result,"p_manage.php"); 
?> 








2. 修改 消息 


消息 发 布 后， 可 以 随时 对 消息 内 容 进行 修改 。 在 消息 管理 页 面 中 ,选择 要 修改 的 消息 列 ， 单 击 “ 修 
改 ” 超 链接 将 进入 消息 修改 页 面 。 在 消息 修改 页 面 中 ， 有 一 个 和 发 布 页 面 非常 相似 的 “修改 ”表单 。 
消息 修改 页 面 中 的 主要 表单 元 素 如 表 3.3 所 示 。 


表 3.3 修改 页 面 的 表单 元 素 








名 称 | 元 素 类 型 重要 属性 含 义 
addmess form action="m _ message_ chk .php" method="post" 修改 消息 表单 
Pp_title text value="<?php echo $rows[1]: ?>" 消息 标题 





P_content textarea <?php echo $rows[2]: ?> 消息 内 容 
<?phpif($rows[4] =— "9"){?> 
<option value="9" selected="selected"> 企 业 公 告 </option> 
<option value="10"> 活 动 安排 </option> 
P_type Select <2phpjelse{f?> 消息 类 型 
<option value="9"> 企 业 公告 </option> 
<option value="10" selected="selected"> 活 动 安排 </option> 











<?php}?> 
id value="<?php echo $rows[0]: ?>" 修改 消息 id 
submit2 value=" 修 改 " onclick="return add_ messO:” “修改 ”按钮 
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消息 修改 完成 后 ， 单 击 “ 修 改 ” 按 钮 ， 在 m_message_chk.php 页 中 将 会 处 理 修 改 事件 ， 通 过 隐藏 
表单 传 过 来 的 id， 定 位 到 要 修改 的 消息 项 ， 使 用 update 语句 实现 对 数据 的 更 新 。 程 序 关 键 代码 如 下 : 


倒 程 09 代码 位 置 : 资源 包 \TM\03\oa\rsxx\m message_chk.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 链接 文件 
+ 更 新 数据 库 */ 


S$sqlstr = "update tb_person set p_title =".$_POST[p_title].",p_content = ".$_POST[p_content].",p_time 
= now(),u_id ="$_POST[p_type]." where id = ".$_POSTT'id"; 

Sresult = mysqli_query($conn ,$sqlstr); 

re_message($result,"p_manage.php"); /1/ 消 息 处 理 函 数 
promi 
?> 





3. 删除 消息 


每 隔 一 段 时 间 管理 员 就 要 定期 清理 一 下 无 用 的 消息 和 活动 安排 ， 以 加 速 页 面 的 浏览 速度 ， 减 轻 服 
务 器 压力 。 删 除 消息 的 运行 结果 如 图 3.20 所 示 。 








图 3.20 删除 消息 的 运行 结果 


删除 消息 时 ， 选 择 要 删除 的 消息 列 ， 单 击 “ 删 除 ” 超 链接 ， 弹 出 确认 删除 对 话 框 ， 单 击 “ 确 定 ” 
按钮 ， 将 要 删除 消息 的 id 号 传 给 系统 ， 通 过 消息 处 理 函数 将 消息 删除 。 关 键 代 码 如 下 : 


倒 程 10 ”代码 位 置 ， 资源 包 \TM\03\oa\rsxx\d_message_chk.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 库 链 接 文件 
include "../inc/func.php"; /包含 函 数 处 理 文件 
/删除 数据 


$sqlstr = "delete from tb_person where id = "$_GETTid]; 
Sresult = mysqli_query($sqlstr, $conn); 
re_message(S$result,"./p_manage.php"); 1/ 消 息 处 理 函 数 


ee 
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3.7.4 意见 箱 的 实现 过 程 


国 意见 箱 模块 使 用 的 数据 表 : tb lyb 


意见 箱 是 用 户 匿名 发 言 的 一 个 留言 板 。 对 于 用 户 的 留言 ， 只 有 人 事 部 的 人 员 才 有 权 回 复 或 删除 ， 


对 于 其 他 部 门 的 员工 则 只 可 以 留言 
和 查看 , 而 无 法 进行 其 他 操作 。 意见 
箱 的 运行 结果 如 图 3.21 所 示 。 

在 本 页 面 中 , 系统 首先 读 取 出 意 
见 箱 数据 表 (tb_ lyb) 中 的 所 有 数据 ， 
并 逐条 显示 , 在 每 条 显示 记录 后 , 有 
一 个 判断 式 , 判断 用 户 所 属 部 门 是 否 
为 人 事 部 , 如 果 是 人 事 部 成 员 , 将 显 
示 “ 回 复 ” 和 “删除 ” 超 链接 ;, 如 果 
不 是 人 事 部 ， 则 不 显示 这 两 个 操作 。 
这 里 是 强制 性 的 , 也 就 是 说 , 即使 是 
管理 员 , 也 无 法 设置 这 里 的 权限 ,只 
有 开发 人 员 才 可 以 改变 。 如 果 想 使 用 
这 种 技术 , 就 一 定 要 确保 这 项 功能 权 
限 不 会 轻易 地 更 改 。 意 见 箱 页 面 的 关 
键 代码 如 下 : 
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图 3.21 意见 箱 页 面 的 运行 结果 


倒 程 11 代码 位 置 : 资源 包 \TM\03\oa\zytd\lyb.php 





<?php 

session_start(); 

include "../conn/conn.php"; 
include "../inc/chec.php"; 
include "../inc/func.php"; 

人 显示 意见 箱 信息 */ 


/开启 session 支持 
// 包 含 数 据 库 链接 文件 
/包含 权限 检查 文件 
// 包 含 自 定义 函数 文件 


$L sql = "select id,|_title,|_content,|_time,is_replyr_back from tb_lyb order by id desc"; 


S$l_rst = mysqli_query($conn, $Lsql); 


while($Lrows = mysqli_fetch_array($l_rst)X{ 
?> 
<tr> 


/开始 循环 输出 每 条 记录 


<td width="15%" height="30" align="left" valign="middle" scope="col"> 


<?php echo $I_rows['\_time']; ?> 

</td> 

<td align="left" valign="middle" scope="col"> 
主题 : <?php echo $L_rows['l_title]; ?> 

</td> 

<!- 判 断 登录 用 户 是 否 为 人 事 部 成 员 --> 


<td width="159%" align="center valign="middle" scope="col"> 
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<?php if(($_SESSION[u_depart]) == "人 事 部 "){ ?> 

<!- 当 登录 用 户 是 人 事 部 成 员 时 ， 显 示 可 执行 操作 -> 

<a href="t_back.php?id=<?php echo $l_rows[id]; ?>"> 回 复 </a>&nbsp;&nbsp;<a href="del_ly_chk.php?id= 
<?php echo $L_rows['id]; ?>" onclick="return del_mess()"> 删 除 </a> 

<?php} 

?>&nbsp; 
</td> 
<I-- 一 一 - 





aa 
</tr><tr> <td height="75" colspan="3" align="left" valign="middle" scope="col" > 
<?php echo $I_rows['l_content]; ?> 
</td></tr> 
<?php 
<!-- 当 字段 is_reply 的 值 为 1 时， 说 阴 有 回复 信息 --> 
if($l_rows['is_reply] == 1X{ 
?> 
<tr> 
<td height="25" colspan="3" align="left" valign="middle" headers="50" scope="col" 
onmouseover="this.style.backgroundColor=#FFEEBC" onmouseout="this.style.backgroundColor=""> 
<font color="#FF0O000"> 
<!-- 显 示 回复 信息 --> 
<?php echo $L_rows[r_back]; ?> 
</font></td> </tr> 
<?php}} 
Pe 
?></table> 
<!-- 留 言 菜单 -> 


一 > 





< 二 





下 面 介绍 匿名 留言 和 回复 、 删 除 留言 的 实现 过 程 。 
1. 匿名 留言 


留言 页 面 主要 是 一 个 HTML 表单 ， 因 为 允许 匿名 留言 ， 所 以 表单 中 没有 隐藏 域 ， 该 页 面 中 的 表单 
元 素 列表 如 表 3.4 所 示 。 


表 3.4 留言 页 面 的 菜单 表单 元 素 表 







重要 属性 




















lyb fom [ia="lyb" name="lyb" method="post" action="lyb_chk php" | ”留言 表单 
1 title text | type="text" name="] title" size="30" | 意见 主题 
1_content textarea | name="] content" cols="75" rows="10" | 意见 内 容 
Submitl hidden | type="reset" value=" 重 置 " | “ 重 置 ”按钮 
Submit submit type="submit" value=" 提 交 " onclick="return add lyb0:" “提交 ”按钮 


当 用 户 填写 完 意见 标题 和 意见 内 容 后 ， 单 击 “ 提 交 ” 按 钮 ， 发 布 意见 ， 系 统 将 转 到 意见 处 理 页 
(lyb_chk.php) 中 对 意见 进行 处 理 : 如 果 意 见 标题 和 意见 内 容 都 不 为 空 , 则 执行 insert 语句 , 向 表 tb_lyb 
中 添加 一 条 新 数据 。 程 序 关键 代码 如 下 : 


® 
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<?php 
session_start(); /开启 session 支持 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 


/* 判 断 输 入 是 否 为 空 ， 如 果 为 空 ， 则 返回 ， 如 果 无 错误 ， 则 执行 insert 语句 添加 新 记录 */ 
if((trim($_POSTTItitle]) {= ") and (tim($_POSTT!_content]) = ™")){ 
$l_sql = "insert into tb_lyb values(",".$_POSTT['_title].™",".$_POST[I_content].",now(),",")"; 
S$l_rst = mysqli_query($conn, $l_sql); 





re_message($l_rst,"lyb.php?u_id=24"); /使 用 自 定义 函数 
aa 时 
} 
else 

echo "<script>alert( 内 容 和 消息 不 允许 为 空 ");history.go(-1);</script>"; 
?> 





2. 回复 、 删 除 留言 操作 


在 意见 箱 页 面 中 ， 当 系统 判断 当前 登录 用 户 为 人 事 部 成 员 后 ， 将 显示 留言 信息 后 面 的 、 可 操作 的 
功能 模块 “回复 ”和 “删除 ”。 单 击 相应 留言 主题 后 面 的 “回复 ” 超 链接 ， 进 入 回复 留言 页 面 。 回 复 留 
言 页 面 的 运行 结果 如 图 3.22 所 示 。 








-OR | 


3.22 回复 留言 页 面 的 运行 结果 
回复 留言 页 面 中 的 主要 表单 元 素 如 表 3.5 所 示 。 
表 3.5 回复 留言 表单 的 主要 元 素 







重要 属性 
action="t_back_chk.php" method="post" | 回复 留言 表单 
Value="<?php echo read field($conn."tb lyb"."] title".$ GET[id]): ?>" 


name="r back" id="r back" cols="75" rows="10" 













回复 内 容 
name="id" value="<?php echo $ GET[id]: ?>" | 回复 消息 id 
value=" 回 复 " onclick="return re_backO:" 








在 回复 留言 页 面 中 ， 当 用 户 输入 回复 信息 后 ， 单 击 “ 回 复 ”按钮 提交 表单 ， 在 表单 处 理 页 (tback 
chk.php ) 中 系统 将 会 根据 回复 信息 ， 更 新 表 记录 。 关 键 代 码 如 下 : 
@ 
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倒 程 12 ”代码 位 置 : 资源 包 \TM\03\oa\zytd\t_back_chk.php 





<?php 

session_start(); /开启 session 支持 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 


/根据 id， 更 新 数据 库 %/ 


$sqlstr = "update tb_lyb set r_back=".$_POST[r_back].",is_reply = 1 where id = ".$_POST['id]; 


$result = mysqli_query($conn, $sqlstr); 
re_message($result,"lyb.php?u_id=24"); 


ee 


?> 


/信息 函数 





对 于 不 需要 再 保留 的 留言 信息 ， 可 将 其 删除 。 单 击 相应 消息 主题 后 面 的 “删除 ” 超 链接 ， 弹 出 确 
认 删 除 提示 对 话 框 ， 单 击 “ 确 定 ”按钮 将 消息 的 id 传 给 系统 ， 通 过 SQL 语句 将 其 删除 。 程 序 关键 代码 





如 下 : 
倒 程 13 ”代码 位 置 : 资源 包 \TM\03\oa\zytd\del ly_chk.php 
<?php 
session_start(); /开启 session 支持 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文 件 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 
$sqlstr = "delete from tb_lyb where id = "$_GET[id]; /删除 信息 的 SQL 语句 
Sresult = mysqli_query($conn, $sqlstr); /执行 语句 
re_message($result,"lyb.php?u_id=24"); // 返 回 结果 


?> 








3.8 考勤 管理 模块 设计 
1 视频 讲解 
考勤 功能 是 办 公 自 动 化 管理 系统 中 每 天 都 要 使 用 到 的 功能 模块 之 一 ， 也 是 比较 重要 的 模块 之 一 。 
除了 包括 正常 的 上 下 班 登记 功能 外 ， 还 包括 病 事假 登记 和 加 班 登记 等 特殊 登记 功能 。 管 理 员 通过 考勤 
记录 实现 对 上 下 班 及 加 班 标准 时 间 的 设置 。 


3.8.1 考勤 管理 模块 概述 


考勤 模块 主要 的 功能 介绍 如 下 。 

回 上 下 班 登记 : 包括 上 班 登记 和 下 班 登记 。 

回 “ 病 事假 登记 : 包括 病假 登记 和 事假 登记 。 

回 “加班 登记 : 包括 加 班 上 班 登记 和 加 班 下 班 登记 。 

回 ”考勤 记录 : 包括 上 下 班 标准 时 间 设置 和 加 班 标准 时 间 设 置 。 
考勤 模块 的 活动 图 如 图 3.23 所 示 。 


@ 
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图 3.23 考勤 管理 活动 图 


3.8.2 ”考勤 管理 模块 技术 分 析 


在 PHP 中 查询 数据 表 中 数据 的 一 般 步 又 如 下 : 

(1) 链接 数据 库 〈 链 接 文件 conn.php)。 

(2) 执行 SQL 语句 ， 返 回 查 询 结 果 集 ($result = mysqli_query($conn, $sqlstr)。 

(3) 配合 while 循环 语句 输出 查询 结果 (while($rows = mysqli_fetch_row($result)))。 

(4) 关闭 数据 库 。 

在 步骤 (3) 中 ， 如 果 要 取得 某 个 字段 的 值 ， 使 用 $rows[$num] 即 可 。 但 通过 图 3.23， 读 者 可 以 发 
现 , 从 开始 登记 到 保存 登记 记录 , 虽然 只 是 一 个 功能 的 实现 , 却 要 进行 至 少 3 次 判断 。 如 果 都 用 “0,1,2…” 
做 数组 下 标 ， 过 一 段 时 间 后 ， 根 本 记 不 起 取得 的 数据 是 什么 、 有 什么 作用 。 在 后 期 测试 和 系统 维护 时 ， 
还 要 逐一 地 查找 数据 表 来 比 对 。 除 了 mysqli_fetch_row0 函 数 外 ，PHP 还 提供 了 其 他 相似 的 函数 ， 下 面 
就 来 学 习 一 下 。 


1. mysqli_fetch_array() 函 数 


mysqli_fetch_array0 函 数 的 用 法 和 mysqli_fetch_row0 函 数 十 分 相似 , 唯一 不 同 的 是 结果 集 的 数组 下 
标 是 所 查找 的 数据 表 的 字段 值 。 
例如 ， 读 取 数 据 表 tb_register 中 的 “登记 时 间 ” 字 段 ， 代 码 如 下 : 





$str = $rows[r_time]; 


这 样 ， 即 使 时 间 间 隔 比较 长 ， 甚 至 换 人 开发 ， 也 能 明白 该 变量 的 含义 。 
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2. mysqli_fetch_object() 函 数 


mysqli_fetch_object0 函 数 与 其 他 两 个 函数 的 最 大 不 同 就 在 于 : 获取 的 结果 集 是 以 对 象 的 形式 存储 
的 ， 而 不 是 数组 的 形式 。 也 就 是 说 ， 在 读 取 字 段 变量 时 所 使 用 的 格式 不 再 是 $rows[$num]， 代 码 如 下 : 


$str = $rows-->r_time; 


下 面 来 看 一 个 实例 ， 分 别 使 用 上 面 3 种 不 同 的 函数 来 获取 数据 表 tb_setup 的 id 值 ， 代 码 如 下 


<?php 
/链接 数据 库 % 
$conn = mysqli_connect("localhost","root","root"); 
mysqli_select_db($conn, "db_office"); 
mysqli_query("set names utf-8"); 
/创建 和 执行 SQL 语句 */ 
$sqlstr = "select * from tb_setup"; 
Sresult = mysqli_query($conn ,$sqlstr); 
/使 用 mysqli_fetch_row() 函 数 输出 字段 值 */ 
Srows_1 = mysqli_fetch_row($result); 
$str = "mysqli_fetch_rows 函数 : ".$rows_1[0]."<br>"; 
/使 用 mysqli_fetch_array() 函 数 输 出 字段 值 */ 
Srows_2 = mysqli_fetch_array($result); 
$str .="mysqli_fetch_array 函数 : ".$rows_2['id]."<br>"; 
/使 用 mysqli_retch_object() 函 数 输出 字段 值 */ 
$rows_3 = mysqli_fetch_object($result); 
$str .= "mysqli_fetch_object 函数 : ".$rows_3->id."<br>"; 
echo "<script>alert(". $str.");</script>"; 





?> 


输出 结果 为 : 


mysqli_fetch_rows 函数 : 1 
mysqli_fetch_array 函数 : 2 
mysqli_fetch_object 函数 : 3 








3.8.3 ”上 下 班 登 记 的 实现 过 程 


国 。 上 下 班 登录 使 用 的 数据 表 : tb_register 

单 击 “考勤 管理 ”选项 中 的 “上 下 班 登记 ” 
子 选项 ， 在 主 显示 区 (mainFrame) 内 显示 个 人 
上 下 班 的 登记 记录 和 登记 链接 ,用 户 可 以 在 这 里 
进行 登记 和 查找 。 上 下 班 登记 页 面 的 运行 结果 如 
图 3.24 所 示 。 

在 上 下 班 登记 页 面 中 ， 通 过 登录 用 户 ia， 从 
数据 表 tb_register 中 返回 用 户 的 登记 记录 集 ， 并 
显示 在 页 面 中 。 在 显示 数据 时 ， 需 要 进行 2 次 判 




















图 3.24 上 下 班 登记 的 运行 结果 
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断 : 一 次 判断 用 户 的 登记 类 型 《上 班 还 是 下 班 ); 另 一 次 判断 用 户 的 登记 状态 〈 迟 到 、 早 退 、 正 点 上 下 
班 )。 关 键 代码 如 下 : 


倒 程 14 代码 位 置 资源 包 \TM\03\oa\kqel\work_note.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 库 链 接 文件 
include "../inc/func.php"; /包含 自 定义 函数 文件 
?> 


<?php include('..top.php") ?> 
<div class="container"> 
<?php include('../left.php') ?> 
<div class="right"> 


©@ <ahref="javascript:;" onclick="workResiger('<?php echo $_GET[u_id]; ?>");"> 
<?php 
Ar* 判 断 是 哪 种 功能 类 型 */ 
© if$_GET['u id] == 14){ 
echo "上 下 班 登 记 "; 
jelse if($_GET['u_id'] == 15){ 
echo " 病 事假 登记 
jelse if($_GET[u_id'] == 16){ 
echo "加 班 登记 "; 





} 
?></a> 
<?php 
”显示 登记 信息 */ 
$sqlstr = "select id,r_date,r_time,r_type,r_state,r_remark from tb_register where r_id = ".$_GET[u_id]." 
and p_id = ".$_SESSION["'id"]; 
Sresult = mysqli_query($conn, $sqlstr); 
while($rows = mysqli_fetch_array($result)}{ 
?> 
<tr> 
<!- 输 出 登记 日 期 -> 
<td height="25" align="center" valign="middle"><?php echo $rows['r_date']; ?></td> 
<|- 输 出 登记 时 间 一 > 
<td height="25" align="center" valign="middle"><?php echo $rows['r_time']; ?></td> 
<td height="25" align="center" valign="middle"> 
< 上 -输出 登记 类 型 --> 
<?php 
© switch($rows['r_type']){ 
case 0: 
echo “下 班 "”; 
break; 
Case 1: 
echo "上 班 "; 
break:; 
Case 2: 
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echo "加 班 签到 "; 
break:; 
Case 3: 
echo "加 班 签 退 "; 
break; 
Case 4: 
echo "病假 "; 
break:; 
case 5: 
echo "事假 "; 
break; 
}?></td> 
< -一 一 -一 一 一 一 > 
<td height="25" align="center" valign="middle"> 
< 上 -输出 登记 状态 --> 
<?php 
9 switch($rows['r_type']}{ 
case 1: 
echo ($rows[r_state] == 0)?" 正 点 上 班 "" 迟 到 "; 
break; 
case 0: 
echo ($rows[r_state] == 0)?" 早 退 "… 正 点 下 班 "; 
break; 
case 2: 
echo ($rows[r_state] == 0)?" 正 点 加 班 …" 晚 点 加 班 "; 
break; 
case 3: 
echo ($rows[r_state] == 0)?" 加 班 早退 ":" 加 班 下 班 "; 
break; 
case 4: 
echo ($rows[r_state] == 0)?" 病 假 ":" 事 假 "; 
break; 
]?> 
<!— 





</td> 
< 上 -输出 备注 -> 
<td height="25" align="center” valign="middle"><?php echo ($rows[r_remark] != null)?$rows 
[r_remark]: 无 ; ?></td> 
</tr> 
<?php }?> 


人 


< 代码 贴 十 
@ onclick="…": 单 击 事件 ， 执 行 workResigerO 函 数 ， 该 函数 使 用 Layer 弹 层 弹出 一 个 窗口 。 
加 根据 id， 也 就 是 参数 U id 的 值 来 判断 是 哪 种 功能 类 型 。 分 3 种 情况 : 上 下 班 登记 、 病 事假 登记 和 加 班 登记 。 
目 判断 登记 类 型 ， 一 共 分 6 种 情况 : 上班、 下班、 病假、 事假、 加 班 签 到 和 加 班 答 退 。 
@ 判断 状态 类 型 ， 一 共 分 10 种 情况 : 上 班 迟 到 、 上 班 早退 、 正 点 上 班 、 正 点 下 班 、 正 点 加 班 、 加 班 下 班 、 加 班 晚 
点 、 加 班 早退 、 病 假 和 事假 。 


他 
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当 弟 市 “上 下 丙 闪 记 ” 超 链 护 咱 总 全 国生 
弹出 登记 页 面 , 用 户 可 以 进行 上 下 班 的 登记 ge 上 PE @ 


操作 ,在 “登记 类 型 ”下拉 列表 框 中 选择 登 。 73a 2 和 于 
记 类 型 ， 如果 是 迟到 或 早退 等 特殊 情况 , 需 。 2 sa 5 
要 在 “备注 ”列表 框 中 说 明 原因 ， 最 后 单 击 
“登记 ”按钮 进行 登记 ， 如 果 登 记 成 功 ， 将 a 
显示 提示 对 话 框 。 登 记 页 面 的 运行 结果 如 
图 3.25 所 示 。 
在 登记 界面 中 , 主要 有 3 个 表单 元 素 和 图 3.25 ”登记 页 面 的 运行 结果 
一 个 隐藏 表单 ， 如 表 3.6 所 示 。 


表 3.6 上 下 班 登记 的 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含 义 
Pl form method="post" action="p_login_chk.php" 登记 表单 
u name text Value="<?php echo $ SESSION[u_name]: ?>" readonly="readonly" 登记 人 


<?php if($_ GETI['r id] 一 14)1> 
<option value="1"> 上 班 </option> 
<option value="0"> 下 班 </option> 
<2php }else if($_GET['r id] — 15){?> 
<option value="4"> 病 假 </option> 











Wype select <option value="5"> 事 假 </option> 登记 关 弄 
<?php }else if($_GET['r id'] = 16){9> 
<option value="2"> 加 班 签 到 </option> 
<option value="3"> 加 班 签 退 </option> 
<?php }?> 
r_remark textarea name="r remark" rows="5" 备注 
rid hidden value="<?php echo $_GET[r id]: ?>" 功能 id 
Submit2 Submit value=" 回 复 " onclick="return re_backO:" “登记 ”按钮 





选择 相应 的 登记 类 型 ， 如 上 班 ， 提 交 表单 后 ， 系 统 就 会 跳 到 处 理 页 (p_login_chk.php) 处理。 在 处 
理 页 中 ， 首 先 判断 登记 类 型 《上 班 、 下 班 )， 然 后 根据 登记 类 型 取得 相对 应 的 标准 时 间 ， 通 过 两 个 时 间 


的 对 比 ， 得 到 登记 状态 迟到、 早退 、 正 点 上 下 班 )， 最 后 执行 insert 语句 ， 添 加 新 记录 。 程 序 关键 代 
码 如 下 : 


倒 程 15 代码 位 置 资源 包 \TM\03\oa\kqel\p_login_chk.php 








<?php 

session_start(); /开启 session 支持 
include "../conn/conn.php"; /包含 数 据 库 链 接 文件 
include "../inc/chec.php"; // 包 含 权限 检查 文件 


A1 代表 迟到 ，0 代表 正点 登记 */ 
/判断 功能 类 型 ， 如 上 下 班 登记 、 加 班 登记 */ 
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if(($r_id == "14") or (Sr_id == "16")X{ 
判断 登记 类 型 ， 如 上 班 、 下 班 等 */ 


if($u_type == 0) 

$t_sql = "select * from tb_setup where id = 2"; // 取 得 下 班 标准 时 间 
else if($u_type == 1) 

$t_sql = "select * from tb_setup where id = 1"; // 取 得 上 班 标准 时 间 
else if($u_type == 2) 

$t_sql = "select * from tb_setup where id = 3"; // 取 得 加 班 上 班 时 间 
else if($u_type == 3) 

$t_sql = "select * from tb_setup where id = 4"; // 取 得 加 班 下 班 时 间 





Pe 1 
St_rst = mysqli_query($t_sql,$conn); 


St_rows = mysqli_fetch_row($t_rst); 


$s_time = $t_rows[2]; // 规 定 的 标准 时 间 
@ $now time= date("H:i:s"); /| 当前 的 登记 时 间 


四 $sql="insert into tb_register (r_date,r_time,r_type,r_state,r_remark,r_id,p_id) 
values(".date("Y-m-d").",".date("H:i:s").",".$_POST[u_type'].",".((strtotime($now_time) - 
strtotime($s_time) > 
0)?1:0).",".$_POSTI'r_remark].",".$_POST[r_id].",".$_SESSIONTid].")"; 
jelse if($r_id == "15") 

$l_sql = "insert into tb_register (r_date,r_time,r_type,r_state,r_remark,r_id,p_id) 
values(".date("Y-m-d").",".date("H:i:s").",".$_POST['u_type].",",".$_POST[r_remark].",".$_POST[r_id].",".$_ 
SESSION[id].")"; 
S$l_rst = mysqli_query($conn, $Lsql); 
上 * 登 记 完成 后 ， 关 闭 窗口 */ 
if($l_rst) 

echo "<script>alert(' 登 记 完成 ');window.close();</script>"; 
else 

echo "<script>alert(' 错 误 ');history.go(-1);</script>"; 

J 


i 





< 代码 贴 十 

@ date0 时 间 函 数 可 以 通过 不 同 的 组 合 返回 不 同 的 时 间 样 式 。 例 如 程序 中 的 date("H:is")， 还 可 以 加 上 年 、 月 、 日 、 
星期 等 ， 如 date("Y-m-d H'is 1")。 

@ ((strtotime($now_time) - strtotime($s_time) > 0)?1:0): 根据 以 上 内 容 得 到 的 SQL 语句 ， 其 中 strtotime 函数 是 PHP 
的 一 个 时 间 函 数 ， 作 用 是 将 一 个 日 期 时 间 描 述 解析 为 一 个 unix 时 间 鹤 ， 如 strtotime("2007-07-11 13:49:29") 输出 的 结果 
就 是 1184132969， 这 时 对 两 个 时 间 进 行 计算 ， 就 可 以 得 到 时 间 的 大 小 关系 。 


3.8.4 设置 时 间 的 实现 过 程 


国 考勤 管理 使 用 的 数据 表 : tb Tegister、tb setup 

考勤 记录 页 面 的 功能 主要 有 两 部 分 : 一 是 显示 所 有 员工 的 考勤 信息 ; 二 是 设置 标准 时 间 。 显 示 所 
有 员工 的 考勤 信息 的 实现 过 程 和 3.8.3 节 中 的 实现 方式 相 类 似 ， 唯 一 不 同 的 就 是 包括 了 全 体 职员 ， 只 要 
略微 改动 SQL 语句 即 可 。 


全 


置 时 间 页 面 ， 如 图 3.26 所 示 。 
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下 面 主要 来 看 一 下 设置 标准 时 间 页 面 。 在 “考勤 记录 ”页 面 中 单 击 “ 设 置 时 间 ” 超 链接 ， 进 入 设 



































在 本 页 面 中 ， 从 tb_setup 表 中 读 取 预 定义 的 标准 时 间 ， 有 4 Fn 一 一 rm 一 
个 标准 时 间 ， 分 别 是 “上 班 签 到 ”“ 下 班 签 退 ” “加班 签 到 ” 本 一 一 要 一 要 一 一 要 
和 “加 班 签 退 ”。 程 序 关键 代码 如 下 : 图 3.26 设置 时 间 页 面 
倒 程 16 代码 位 置 资源 包 \TM\03\oa\kqel\set_time.php 
<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检测 文件 
include "../conn/conn.php"; /包含 数据 库 链接 文件 
?> 


<?php include("../top.php') ?> 
<div class="container"> 
<?php include('../left.php') ?> 
<div class="right"> 
<form name="form1" method="post" action="set_time_chk.php"> 
<table border="1" cellspacing="0" cellpadding="0" background=". /images/bg.jpg"> 
<tr> 
<td width="150" height="25" align="center" valign="middle"> 上 班 签 到 </td> 
50" height="25" align="center" valign="middle"> 下 班 签 退 </td> 
"150" height="25" align="center" valign="middle"> 加 班 签 到 </td> 
<td width="150" height="25" align="center" valign="middle"> 加 班 签 退 </td> 





</tr> 
<?php 
S$sqlstr = "select * from tb_setup"; // 提 取 表 中 的 标准 时 间 
Sresult = mysqli_query($conn, $sqlstr); /| 返回 结果 集 
$num = 0; // 设 置 序号 
?> 
<tr> 


<?php while($rows = mysqli_fetch_row($result)){ ?> 
<td height="30" align="center" valign="middle"> 
©@ <inputtype="text" name="| time<?php echo $num;?>" value="<?php echo $rows[2]; ?>" size=15></td> 
<?php $num++;}?> 
</tr> <tr> 
© <tdheight="30" align="center" valign="middle"><input type="submit" name="u_logo" value=" 设 置 "></td> 
<td height="30" align="center" valign="middle"><input type="submit" name="d_logo" value=" 设 置 "></td> 
<td height="30" align="center" valigl iddle"><input type="submit" name="a_logo" value=" 设 置 "></td> 
<td height="30" align="center" valign="middle"><input type="submit" name="q_logo" value=" 设 置 "></td> 
</tr> 
</table> 
</form> 





亏 人 代码 贴 二 


@ 变量 Snum 是 为 了 在 while 循环 语句 中 输出 不 同 的 表单 元 素 的 名 称 ， 每 次 循环 $num 自 加 1。 
@ 生成 4 个 设置 时 间 的 按钮 ， 在 处 理 页 中 ， 根 据 不 同 的 名 字 做 不 同 的 处 理 。 
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当 调 整 了 其 中 的 一 项 后 ， 单 击 “ 设 置 ”按钮 ， 进 入 到 处 理 页 ， 在 处 理 页 中 ， 会 根据 按钮 的 名 称 和 
text 元 素 的 value 值 来 进行 相关 的 操作 。 程 序 关键 代码 如 下 : 


倒 程 17 代码 位 置 资源 包 \TM\03\oa\kqel\set_time_chk.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 库 链接 文件 


/根据 名 称 判断 事件 是 哪个 按钮 产生 的 ， 并 生成 不 同 的 SQL 语句 */ 
if(isset($_POSTI[u_logo])X{ 

$sqlstr = "update tb_setup set | time =".$_POSTILtime0]. " where id = 1"; // 设 置 上 班 时 间 
jelse if(isset($_POST[d_logo])X{ 

$sqlstr = "update tb_setup set | time =".$_POST[Ltime1]. "where id = 2"; // 设 置 下 班 时 间 
jelse iflisset($_POST[a_logo])){ 

$sqlstr = "update tb_setup set |_time =".$_POSTI[Ltime2]. " where id = 3 ”// 设 置 加 班 上 班 时 间 
jelse iflisset($_POST[q_logo])) 

$sqlstr = "update tb_setup set |Ltime =".$_POSTILtime3]. " where id = 4"; // 设 置 加 班 下 班 时 间 
} 
else 

echo "<script>alert(' 非 法 登录 ');window.close();</script>"; 
J 
Sresult = mysqli_query($conn, $sqlstr); 
if($result) 

echo "<script>alert(' 设 置 成 功 ');window.close();</script>"; 
else 

echo "<script>alert(' 设 置 失 败 ');window.close();</script>"; 





3.9 后台 首页 设计 





| 


由 于 后 台 管理 主要 是 针对 管理 员 开放 的 ， 普 通用 户 无 法 访问 到 这 里 ， 所 以 在 页 面 风格 的 设计 上 相对 
要 简单 得 多 。 不 需要 艳丽 的 图 片 、 极 炫 的 特效 ， 只 要 干净 、 整 洁 ， 给 人 一 种 朴实 无 华 的 感觉 是 最 好 的 。 


3.9.1 后 台 首 页 概述 


如 果 说 前 台 是 系统 的 外 衣 ， 那 么 后 台 就 是 系统 的 灵魂 。 数 据 更 新 、 系 统 升级 ， 甚 至 是 安全 隐患 ， 
都 源 于 后 台 的 设计 是 否 完善 ， 功 能 是 否 强大 。 可 以 说 ， 后 台 越 完美 ， 系 统 拥有 的 外 衣 才 会 越 华丽 ， 人 
们 的 目光 才 会 被 吸引 。 

办 公 自 动 化 管理 系统 的 后 台 首页 主要 包括 的 功能 如 下 。 

回 ”网 站 首部 导航 栏 ; 包括 当前 时 间 、 首 页 、 重 新 登录 和 退出 登录 。 

回 ”网 站 左 侧 导 航 栏 : 包括 各 个 管理 模块 及 分 类 。 

回 ”部 门 管理 模块 : 主要 用 于 对 部 门 信息 的 操作 ， 包 括 添 加 、 修 改 、 删 除 和 查看 功能 。 
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加 ”职员 管理 模块 :主要 用 于 对 职员 信息 的 统一 管理 ， 包 括 添加 、 修 改 、 删 除 和 查找 职员 。 

加 ”权限 管理 模块 ， 主 要 用 于 对 部 门 、 职 员 和 用 户 组 的 功能 权限 分 配 。 

回 ”系统 管理 模块 : 维护 系统 日 志 、 修 改 管理 员 密码 和 系统 数据 的 备份 。 

本 案例 中 提供 的 后 台 首页 如 图 3.27 所 示 。 该 
首页 在 本 书 资 源 包 中 的 路 径 为 TMWO3\adminvadmin 
main.php。 


3.9.2 后 台 首 页 技术 分 析 


后 台 页 面 和 前 台 页 面 的 布局 是 相似 的 , 都 是 采 
用 “上 方 固定 ， 左 侧 嵌 套 ”的 框架 结构 。 在 上 方 框 
架 (topFrame) 显示 Banner 和 首部 导航 ， 在 左 侧 
框架 (leftfFrame) 中 显示 功能 列表 。 中 间 的 显示 区 
(manFrame) 显示 各 级 窗 体 。 图 3.27 办 公 自动 化 管理 系统 后 台 首 页 








3.9.3 后台 首 页 的 实现 过 程 


国 ” 后台 首页 使 用 的 数据 表 : tb_controller 
下 面 来 看 一 下 后 台 页 面 中 框架 布局 的 应 用 ， 程 序 代 码 如 下 : 


倒 程 18 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\admin_main.php 


<!-- 定 义 顶 部 框架 和 框架 内 显示 的 内 容 --> 
<frameset rows="88,*" cols="*" frameborder="no" border="1" framespacing="0"> 
<frame src="top.php" name="topFrame" scrolling="No" noresize="noresize" id="topFrame" title="topFrame" /> 
<!- 定 义 嵌 套 的 左 侧 框架 和 框架 内 显示 的 内 容 -> 
<frameset rows="*" cols="216,*" framespacing="0" frameborder="no" border="1"> 
<frame src="leftphp" name="leftFrame" scrolling="No" noresize="noresize" id="leftFrame'" title="leftFrame" /> 
< 定义 主 框架 内 容 --> 
<frame src="main.php" name="mainFrame" id="mainFrame" title="mainFrame" /> 
</frameset> 
</frameset> 
<!-- 当 浏览 器 不 支持 框架 时 所 显示 的 内 容 --> 
<noframes><body> 
</body> 
</noframes></html> 











3.10 ”部 门 管理 模块 设计 














一 个 成 熟 、 稳 定 的 企业 一 定 要 有 一 个 合理 的 部 门 结构 ， 包 括 每 个 部 门 需要 做 什么 
工作 、 部 门 之 间 的 协调 关系 和 部 门 之 间 的 上 下 级 关系 等 ， 都 要 明确 下 来 ， 这 样 才能 各 尽 其 职 ， 不 会 出 
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现 资源 浪费 ， 提 高 企业 整体 的 效率 。 
3.10.1 部门 管理 模块 概述 


部 门 管理 模块 包括 查看 部 门 和 添加 部 门 两 大 部 分 ， 下 面 分 别 介绍 。 
1. 查看 部 门 


查看 各 个 部 门 名 称 和 部 门 之 间 的 上 下 级 关系 ， 还 可 以 对 部 门 的 信息 和 级 别 进行 修改 ， 如 果 是 最 底 
层 的 部 门 ， 则 可 以 对 其 进行 删除 操作 。 


2. 添加 部 门 


可 以 添加 新 的 部 门 ， 包 括 部 门 名 称 、 上 级 部 门 和 部 门 备注 。 
部 门 管理 框架 如 图 3.28 所 示 。 


避 
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删除 成 功 














图 3.28 部 门 管理 操作 流程 图 
3.10.2 ”部 门 管理 模块 技术 分 析 


部 门 管理 中 的 技术 难点 是 : 如 何 实现 查看 部 门 模块 中 的 动态 显示 部 门 结构 的 功能 。 这 里 用 
JavaScript+ 递 归 函 数 来 实现 。 下 面 详 细 讲 解 。 

当 用 户 单 击 “ 查 看 部 门 ” 时 ， 系 统 开始 查找 tb_depart 数据 库 。 首 先 找到 根部 门 的 信息 列表 ， 并 判 
断 第 一 个 根部 门 是 否 有 下 级 部 门 。 如 果 没 有 ， 则 直接 输出 部 门 名 称 、 修 改 和 删除 操作 ， 然 后 查找 下 一 
个 根部 门 ;， 如 果 有 下 级 部 门 ， 则 调用 自 定义 函数 list_menu($num,$wid,&$m)。 其 中 : 

回 $num: 为 上 级 部 门 id 号 。 

回 $wid: 为 表格 宽度 。 


全 
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回 $m: 为 表单 元 素 id 变量 值 。 

在 这 个 函数 中 ， 执 行 步骤 如 下 : 

(1) 查找 上 级 部 门 为 $num 的 部 门 列表 。 

(2) 判断 第 一 个 部 门 是 否 有 下 级 部 门 。 

(3) 如 果 没有 ， 则 输出 部 门 名 称 、 修 改 和 删除 操作 ， 然 后 查找 下 一 个 同 级 部 门 。 

(4) 如 果 有 下 级 部 门 ， 则 再 次 调用 list_menu($num,$wid,&$m) 函 数 。 

(5) 重复 步骤 (1) 一 〈4)。 

这 时 ， 相 信 读 者 已 经 看 出 : list menu0 正 是 一 个 递归 函数 。 递 归 函 数 就 是 在 当 满 足 一 定 的 条 件 时 ， 
函数 自己 调用 自己 ， 经 过 层 层 调用 ， 最 终 达到 解决 问题 的 目的 。list_menu0 函 数 的 代码 如 下 : 


倒 程 19 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\inc\func.php 


<?php 
function list_menu($num,$wid,&$m){ 
上 * 连 接 数据 库 */ 
S$conn = mysqli_connect("localhost","root","root"); 
mysqli_select_db($conn ,"db_office"); 
mysqli_query("set names utf8"); 
[J 
A 查询 同 级 部 门 */ 
$sqlstr = "select * from tb_depart where up_depart = ".$num; 
Sresult = mysqli_query($conn, $sqlstr); 


eisai 





?> 
< 上 -隐藏 域 -> 
<tr id="OpenMe<?php echo $m; ?>" style="display:none;"> 
<td> <table width="<?php echo $wid; ?>%" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td height="25" align="center"><table width="<?php echo ($wid -3); ?>%" border="0" cellspacing="0" 
cellpadding="0"> 
<?php 
人 循环 输出 同 级 部 门 */ 
while($rows = mysqli_fetch_row($result)X{ 
hr* 查 看 下 属 部 门 */ 
$sqlstr1 = "select * from tb_depart where up_depart = ".$rows[0]; 
Sresult1 = mysqli_query($conn, $sqlstr1); 
$nu = mysqli_num_rows(S$result1); 
## 当 前 部 门 没有 下 属 部 门 时 
if(I$nu){ 
$m += 1; 
?> 
@ <tronMouseOver="this.style.background='#96F7F4"" onMouseOut="this.style.background=""> 
<td width="100%" align="left">&nbsp;&nbsp;&nbsp;<img src="../images/folder.gif" width="16" height= 
"16" border="0"><?php echo $rows[1] ?> 一 一 一 一 一 一 <a href="edit_depart.php?id=<?php echo $rows[0]; ?>"> 
修改 </a>||<a href="del_depart_chk.php?id=<?php echo $rows[0]; ?>" onClick="return cfm();"> 删 除 </a></td></tr> 
<?php 


} 
上 当前 部 门 有 下 属 部 门 时 沁 
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else{ 
$m+=1; 
?> 
<!- 鼠 标 划 过 样式 --> 
<tr onMouseOver= "this.style.background=#96F7F4" onMouseOut="this.style.background="> 
2 <td>&nbsp;<a href="Javascript:ShowMe(img<?php echo $m; ?>,OpenMe<?php echo $m; ?>)"> 


<img src="../Images/jia.gif" border="0" alt=" 展 开 " id="img<?php echo $m; ?>"><?php echo $rows[1];?> 





</a>--- -- 一 -<a href="edit_depart.php?id=<?php echo $rows[0]; ?>"> 修 改 </a>|| 删 除 </td></tr> 
<?php 
© list_menu($rows[0],$wid,$m); 
» 
?> 
</table></td></tr></table></td></tr> 
<?php}?> 
二 代码 贴 十 


@ onMouseOver…onMouseOut: onMouseOver 表示 鼠标 指针 停留 在 当前 元 素 上 时 的 状态 ，onMouseOut 事件 表示 鼠 
标 指针 离开 当前 元 素 时 的 状态 。 本 行 用 不 同 的 颜色 表示 两 种 不 同 的 事件 。 

@ 当前 部 门 有 下 属 部 门 时 ， 单 击 图 片 或 文字 链接 ， 触 发 一 个 js 事件 ， 显 示 隐 藏 域 。 

四 调用 自身 函数 。 


在 这 个 模块 中 ,同时 使 用 了 另 一 个 自 定义 函数 isbool($id,$dname), 该 函数 将 $dname 参数 和 tb_depart 
表 中 的 部 门 名 称 一 一 比较 ， 如 果 重 复 ， 则 返回 tue， 和 否则 返回 false。Sid 是 在 修改 部 门 时 ， 先 去 掉 要 修 
改 的 部 门 的 名 称 再 进行 比较 。 在 信息 处 理 页 中 ,根据 返回 的 值 进行 操作 。isbool0 函 数 的 程序 代码 如 下 : 


倒 程 20 代码 位 置 ， 资源 包 \TM\03\oa\admin\inc\func.php 





function isbool($id,$dnamej{ 
上 * 连 接 数据 库 */ 
$conn = mysqli_connect("localhost","root", "root"); 
mysqli_select_db($conn ,"db_office"); 
mysqli_query("set names utf8"); 


LJ 
$sqlstr = "select * from tb_depart where id != $id d_name = '$dname'"; /创建 查询 SQL 语句 
Sresult = mysqli_query($conn, $sqlstr); /执行 SQL 语句 
if(mysqli_num_rows($result)>0) /| 判断 是 否 有 记录 集 

Sisbool = true; // 如 果 有 ，S$isbool = true 
else 

$isbool = false; /否则 $isbool = false 
return $isbool; // 返 回 $isbool 


3.10.3 ”部 门 查看 的 实现 过 程 


国 部门 查 看 模块 使 用 的 数据 表 : tb depart 
通过 选择 导航 栏 中 的 “部 门 管理 ”/“ 查 看 部 门 ”选项 ， 进 入 部 门 查看 页 面 。 在 部 门 查看 页 面 中 显 
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在 部 门 查看 页 面 中 ， 使 用 到 了 递归 函数 。 首 E 


先 查 找 根 部 门 ( 根 部 门 的 条 件 是 top_depart = 0)， 
并 返回 根部 门 的 结果 集 ; 接着 使 用 while 循环 输 
出 所 有 根部 门 ， 每 输出 一 个 根部 门 时 ， 都 要 查看 
当前 部 门 是 否 有 子 部 门 ( 下 属 部 门 )。 如果 没有 子 
部 门 ,那么 直接 输出 部 门 名 称 和 对 部 门 的 操作 ( 修 
改 和 删除 ); 如 果 根 部 门 有 子 部 门 , 则 调用 递归 函 
数 〈list menu0)。 该 页 面 的 程序 代码 如 下 : 


倒 程 21 ”代码 位 置 : 资源 包 \TM\03\oa\admin\bmgl\show_depart.php 





人 人事 和 一 一 一 一 一 信用 
-On 全 可 || 际 
- 信 澡 | 时 除 











3.29 部门 查看 页 面 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数 据 库 链接 文件 
include "../inc/func.php"; /包含 自 定义 函数 文件 

?> 

<script src="../js/admin_js.js"></script> JI 引入 js 脚本 文件 

<link href="../css/style.css" rel="stylesheet"> /引入 外 部 css 样式 

<body onLoad="ShowMe(img1,OpenMe1)"> /lonLoad 事件 ， 表 示 当 页 面 被 载 入 时 调用 ShowMe() 方 法 

<!- 查 看 部 门 信息 --> 

<table width="100%" height="25" border="0" cellpadding="0" cellspacing="0"> 

<?php 
A* 找 到 根部 门 */ 


S$sqlstr = "select * from tb_depart where top_depart = 0"; 
Sresult = mysqli_query($conn, $sqlstr); 
/隐藏 域 id 号 */ 


S$m =1; 
上 * 循 环 输出 所 有 根部 门 六 
while($rows = mysqli_fetch_row($result)X{ 
S$wid = 100; 
A* 查 看 下 属 部 门 */ 
$sqlstr1 = "select * from tb_depart where up_depart = ".$rows[0]; 
Sresult1 = mysqli_query($sqlstr1,$conn); 
Snu = mysqli_num_rows($result1); 
上 * 当 前 部 门 没有 下 属 部 门 时 */ 
if(!$nuX{ 
?> 
<tr onMouseOver="this. style.background=#96F7F4" onMouseOut="this.style.background=""> 
<td>&nbsp;&nbsp;&nbsp;&nbsp;<a href="<?php echo $PHP_SELF; ?>"><img src="../Images/folder.gif" 
border="0" alt=" 展 开 " id="img<?php echo $m; ?>"><?php echo $rows[1]; ?></a>- 一 一 一 一 一 <a href="edit_ 
depart.php?id=<?php echo $rows[0]; ?>"> 修 改 </a>||<a href="del_depart_chk.php?id=<?php echo $rows[0] ?>" 
onClick="return cfm():"> 删 除 </a></td> 
</tr> 
<?php 
人 * 当 前 部 门 有 下 属 部 门 时 */ 
}else{ 
?> 


_ 国 ) 
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<tr onMouseOver= "this.style.background=#96F7F4" onMouseOut="this.style.background=""> 
<td><a href="Javascript:ShowMe(img<?php echo $m; ?>,OpenMe<?php echo $m; ?>)"><img src= 

"../Images/jia.gif" border="0" alt=" 展 开 " id="img<?php echo $m; ?>"><?php echo $rows[1]; ?></a>- 一 一 一 一 一 <a 
href="edit_depart.php?id=<?php echo $rows[0]; ?>"> 修 改 </a>|| 删 除 </td> 

</tr> 
<?php 
/输出 同 级 部 门 ， 调 用 递归 函数 

list_menu($rows[0],$wid,$m); 

$m += 1; 


分 
循环 结束 */ 


?> 


1. 修改 部 门 


在 查看 部 门 页 面 中 ， 可 以 对 部 门 的 名 称 、 上 级 部 门 及 主要 职责 等 相关 信息 进行 修改 。 单 击 要 修改 
部 门 名 称 后 面 的 “修改 ” 超 链接 ， 进 入 修改 部 门 页 面 ， 如 图 3.30 所 示 。 


办 公 自 动 化 








图 3.30 ”修改 部 门 页 面 
修改 部 门 页 面 (edit_departphp) 中 的 主要 表单 元 素 如 表 3.7 所 示 。 


表 3.7 修改 部 门 页 面 中 的 表单 元 素 
名 称 | 元素 类 型 重要 属性 含义 
Text size="20" value="<?php echo $rows[d_name] ?> 部 门 名 称 
<?php 
while($tmprows = mysqli fetch array($tmpresult){ 
if($tmprows[id] 一 $rows[top_depart]){ 
echo "<option value=". $tmprows[id]." 





d name 














wit se selected>".$tmprows[d_name]."</option>"; 列 出 所 各 病 站 
jelse 
echo "<option value=". $tmprows[id].">".$tmprows[d_name]."</option>": 
}?> 
remark textarea <?php echo $rows[remark] ?> 备注 信息 
id hidden value="<?php echo $rows[id] ?>" 修改 部 门 的 id 
Submit Submit name="Submit" value=" 修 改 " onclick="return a_checkO:" “修改 ”按钮 
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当 管 理 员 修改 完 相 应 的 信息 后 单 击 “ 修 改 ”按钮 提交 修改 信息 ， 系 统 转 到 处 理 页 (edit_depart_ 
chkphp)， 首 先 判断 输入 的 部 门 名 称 是 否 重复 。 如 果 重 复 ， 则 返回 ; 如 果 不 重复 ， 则 继续 执行 ， 确 定 修 
改 后 的 上 级 部 门 和 根部 门 。 关 键 代码 如 下 : 

倒 程 22 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\bmel\edit_depart_chk.php 


地 全 





session_start(); /开启 session 支持 

include "../inc/chec.php"; // 包 含 权限 检查 文件 

include "../conn/conn.php"; // 包 含 数据 库 链 接 文 件 

include "../inc/func.php"; // 包 含 数据 库 链 接 文 件 
/* 判 断 输 入 部 门 名 称 是 否 重复 */ 


if(isbool($_POST[id'],$_POST[d_name'])}{ 
echo "<script>alert(' 名 称 已 存在 ， 请 重新 输入 !);history.go(-1);</script>"; 
exit(); 


} 
上 * 修 改 部 门 ， 确 定 上 级 部 门 和 根部 门 */ 
if($_POST[u_id] != OX 
上 * 查 找 上 级 部 门 的 根部 门 */ 
$sqlstr = "select top_depart from tb_depart where id = ".$_POST[u_id]; 
Sresult = mysqli_query($conn, $sqlstr); 
S$rows = mysqli_fetch_array($result); 
上 * 如 果 存 在 上 级 部 门 ， 就 取得 上 级 部 门 的 根部 门 */ 
if ($rows[top_depart'] != 0) 
S$top_depart = $rows[top_depart]; 
上 * 如 果 不 存在 ， 将 上 级 部 门 作为 根部 门 */ 
else 
Stop_depart = $_POST[u_id]; 


jelse 
上 * 如 果 没有 上 级 部 门 ， 那 么 就 将 自身 定 为 根部 门 */ 
S$top_depart = 0; 
上 * 创 建 、 执 行 修改 部 门 的 SQL 语句 */ 
$sqlstr = "update tb_depart set d_name = ".$_POST[d_name'].",top_depart = ".$top_depart.", up_depart 


"$_POST[u_id].", remark = ".$_POST[remark']." where id = ".$_POSTTid]; 
Sresult = mysqli_query($conn, $sqlstr); 
re_message($result,"show_depart.php"); 

?> 





2. 删除 部 门 


如 果 要 取消 某 个 部 门 ， 可 以 将 该 部 门 删除 。 在 删除 部 门 时 ， 只 能 对 最 底层 的 部 门 进行 删除 ， 如 果 
部 门下 面 有 子 部 门 ， 其 部 门 后 面 的 “删除 ” 超 链 接 不 可 用 。 删 除 功能 的 实现 相对 比较 简单 一 点 ， 单 击 
“删除 ” 超 链 接 后 ， 系 统 提示 “是 否 确认 删除 部 门 >， 如 果 单 击 “ 确 认 ” 按 钮 ， 系 统 将 转 到 处 理 页 (del_ 
depart_chk.php) 进行 删除 处 理 。 程 序 代 码 如 下 : 


倒 程 23 代码 位 置 资源 包 \TM\03\oa\admin\bmel\del_ depart_chk.php 


<?php 
session_start(); /开启 session 支持 








全 


PHP 项目 开发 全 程 实 录 (第 4 版 ) 





include "../inc/chec.php"; /包含 权限 查看 文件 
include "../conn/conn.php"; // 包 含 数据 库 链接 文件 
include "../inc/func.php"; // 包 含 自 定义 文件 
$sqlstr = "delete from tb_depart where id = "$_GET[id]; /删除 SQL 语句 
Sresult = mysqli_query($conn, $sqlstr); /! 执 行 SQL 语句 
re_message(S$result,"show_depart.php"); // 返 回 最 后 结果 

Ec 





3.10.4 部 门 添加 的 实现 过 程 








用 户 单 击 导航 栏 中 的 “部 门 管理 "/“ 添 。 有 == 
加 部 门 ” 选 项 ， 进 入 添加 部 门 页 面 (add_ 
departphp)， 该 页 面 用 于 设置 新 添加 部 门 的 相 | op 


关 信 息 ,， 包括 部 门 名 称 、 上 级 部 门 和 备注 ， 如 sm rE 

sss 下 aa | 

3.31 所 示 。 aa 日 
添加 部 门 页 面 (add_depart) 主要 包含 一 个 加 | 

添加 部 门 表 单 ， 表 单元 素 的 名 称 、 说 明 及 属性 


值 如 表 38 所 示 图 3.31 添加 部 门 页 面 
不 。 


表 3.8 添加 部 门 页 面 的 表单 元 素 


<?php 
‘while($rows = mysqli_ fetch_row($resultl)) 
echo "<option value=".$rows[0].">".$rows[1]."</option>": 






列 出 所 有 部 门 
?> 


name="remark" cols="40" rows="5" id="remark" 
name="Submit" value=" 添 加 " onclick="return a_checkO:" 


当 管 理 员 提交 表单 后 ， 进 入 处 理 页 (add_depart_chk.php〉 进 行 添加 操作 。 在 处 理 页 中 ， 首 先 判 断 
输入 的 部 门 名 称 是 否 重复 ， 如 果 名 称 符合 要 求 ， 则 进一步 确定 上 级 部 门 和 根部 门 ， 最 后 执行 msert 语句 
进行 添加 操作 。 处 理 页 的 代码 如 下 : 


倒 程 24 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\bmel\add_depart_chk.php 





<? 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 库 链接 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 
让 判断 输入 部 门 名 称 是 否 重复 */ 


iflisbool($_POST[d_name')}{ 
echo "<script>alert(' 名 称 已 存在 ， 请 重新 输入 !!");history.go(-1);</script>"; 
exit(); 


@ 
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} 
确定 上 级 部 门 和 根部 门 六 
if$_POST[u id] = "0"X 
$sqlstr = "select top_depart from tb_depart where id = "$_POST[u_id]; 
Sresult = mysqli_query($conn, $sqlstr); 
Srows = mysqli_fetch_row($result); 
if ($rows[top_depart] = 0) 
S$top_depart = $rows[top_depart]; 
else 
Stop_depart=$_POST[u_id]; 
} 
else 
S$top_depart = 0; 
添加 新 部 门人 
$sqlstr = "insert into tb_depart 
values(",".$_POST[d_name'].",".$top_depart.",".$_POST[U_id].",".$_POST[remark].")"; 
Sresult = mysqli_query($conn, $sqlstr); 
/输出 添加 结果 六 
re_message($result,"show_depart.php") 
?> 





3.11 系统 管理 模块 设计 





系统 管理 模块 在 后 台 系 统 中 占据 着 非常 重要 的 位 置 ， 后 台 其 他 的 模块 都 是 针对 前 台 
功能 而 设计 的 。 而 系统 管理 是 针对 系统 本 身 的 一 些 信息 、 情 况 而 设计 的 ， 一 般 包括 数据 备份 、 管 理 员 信 
息 修改 等 ， 甚 至 可 以 对 操作 系统 的 配置 文件 进行 操作 ， 可 以 说 是 管理 员 的 管理 员 ， 是 后 台 系统 中 必 不 可 
少 的 组 成 部 分 。 


3.11.1 系统 管理 模块 概述 


系统 管理 模块 包括 如 下 功能 。 
加 ”系统 日 志 : 记录 了 前 、 后 台 用 户 登录 时 的 相关 信息 。 
回 数据 备份 : 可 以 对 整个 数据 库 进 行 备份 、 


恢复 和 删除 。 
回 ”修改 密码 :修改 后 台 管理 员 登 录 密 码 。 7 


ee 
系统 管理 用 例 图 如 图 3.32 所 示 。 EE 
CBS 从 DO 一 > 
3.11.2 ”系统 管理 模块 技术 分 析 管理 员 ap> 
在 系统 管理 模块 中 ， 保 存 数据 的 方式 使 用 了 数 图 3.32 系统 管理 用 例 图 





据 库 和 文件 ， 关 于 数据 库 技术 已 经 介绍 很 多 了 ， 本 
节 主 要 讲解 一 下 文件 及 文件 夹 的 相关 操作 。 
对 文件 的 操作 主要 分 3 步 : 打开 文件 、 读 取 / 写 入 文件 和 关闭 文件 。 
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1. 打开 文件 
打开 文件 使 用 的 是 fopen0 函 数 ， 该 函数 的 一 般 格式 如 下 : 
int fopen(string flename,string mode); 


参数 filename 是 要 打开 的 包含 路 径 的 文件 名 ， 可 以 是 相对 路 径 或 绝对 路 径 。 如 果 没 有 任何 前 缀 ， 
则 表示 打开 的 是 本 地 文件 ; 参数 mode 是 打开 文件 的 方式 , 常用 的 有 a (追加 )、w (只 写 ) 和 T (只 读 )。 
代码 如 下 : 


$f_open = fopen("../bak/remark.sql","a+"); 





2. 读 取 / 写 入 文件 


读 取 文件 的 函数 有 很 多 ， 最 常用 的 就 是 按 行 读 取 fgets0 函 数 ) 和 全 部 读 取 (readfile() 函 数 和 
fpassthru() 函 数 )。 
fgets0 函 数 的 一 般 格式 如 下 : 


string fgets(int filename [,int length]); 


参数 filename 是 被 打开 的 文件 ， 参 数 length 是 要 读 取 的 数据 长 度 。 该 函数 能 够 实现 从 filename 指 
定 的 文件 中 读 取 一 行 并 返回 长 度 最 大 值 为 length-1 个 字 节 的 字符 串 。 在 遇 到 换行 符 、EOF 或 者 读 取 了 
length-1 个 字 节 后 停止 。 如 果 没 有 指定 length 的 长 度 ， 默 认 值 是 1KB。 代 码 如 下 : 
<?php 
$f_open = fopen("../bak/remak.txt","r") ; /打开 文本 


echo fgets($f_open) ; // 读 取 其 中 的 数据 
fclose($f_open) ; 








readfile0 函 数 的 一 般 格式 如 下 : 


int readfile($filename); 


参数 $filename 为 要 打开 的 文件 名 ， 使 用 该 函数 时 ， 无 须 进 行 第 1、3 步 ， 也 无 须 输出 语句 echo， 
直接 就 可 以 读 取 文 件 的 全 部 内 容 并 显示 在 页 面 中 。 


3. 关闭 文件 
完成 对 文件 的 操作 后 ， 就 可 以 关闭 文件 了 。 关 闭 文件 的 命令 十 分 简单 ， 格 式 如 下 : 
fclose($open); 


其 中 参数 gopen 是 要 关闭 的 文件 资源 。 
在 本 模块 中 ， 还 用 到 了 两 个 自 定义 函数 c_log0 和 show_file0。c_log0 函 数 的 作用 是 删除 日 志文 件 ， 
程序 代码 如 下 : 


倒 程 25 代码 位 置 ， 资源 包 \TM\03\oa\admin\inc\func.php 
function c¢_log(X{ 


Sfilename="../log.txt"; /日 志文 件 路 径 
@ 
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if(file_exists($filename)) 
unlink($filename); 
else 


1/ 判断 文件 是 否 存在 
// 如 果 文件 存在 ， 删 除 
// 如 果 文件 不 存在 ， 返 回 上 层 操 作 


echo "<script>alert(' 暂 无 系统 日 志 ! ');history.go(-1);</script>"; 


} 


show_file() 函 数 的 作用 是 返回 一 个 目录 下 的 所 有 文件 名 列表 ， 并 以 数组 的 形式 返回 。show_file0 函 


数 的 程序 代码 如 下 : 


倒 程 26 代码 位 置 ， 资源 包 \TM\03\oa\admin\inc\func.php 


function show_file(\{ 
S$folder_name = "../bak"; 
$d_open = opendir($folder_name); 
Snum = 0; 
while($file = readdir($d_open)X{ 
Sfilename[$num] = $file; 
Snum++; 


} 
closedir($d_open); 
return $filename; 


} 


/数据 备份 目录 
/打开 备份 目录 
/数组 下 标 
/循环 读 取 文件 

/将 文件 名 存 为 数组 


/关闭 目录 
// 返 回 文件 数组 





3.11.3 ”系统 日 志 的 实现 过 程 


通过 导航 栏 中 的 “系统 管理 ”/“ 系 统 日 志 ” 可 进入 系 
统 日 志 页 面 。 在 该 页 面 中 显示 系统 日 志 信息 ， 在 系统 日 志 
中 ， 记 录 着 普通 用 户 和 后 台 管 理 员 所 有 人 员 登 录 系统 时 的 
信息 ， 包 括 登 录 时 间 、 用 户 账号 、 登 录 下 等 。 这 些 信息 是 
在 用 户 登录 时 系统 自动 添加 到 数据 表 中 的 (关于 用 户 登录 
的 实现 请 参见 本 书 附带 的 资源 包 )。 系统 日 志 页 面 的 运行 结 


果 如 图 3.33 所 示 。 


显示 日 志 是 对 文件 进行 读 取 ， 定 位 到 日 志文 件 log.txt 
后 ， 使 用 人 etsO 函 数 读 取 日 志 内 容 ， 并 显示 在 页 面 中 。 程 


序 代码 如 下 : 











图 3.33 系统 日 志 页 面 的 运行 结果 


倒 竹 27 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\xtgl\slog.php 





<?php 
session_start(); 
include "../inc/chec.php"; 
?> 
<script src="../js/ladmin_js.js"></script> 
<link href="../css/style.css" rel="stylesheet"> 


<tr> 


/开启 session 支持 
/包含 权限 检查 文件 


/引入 js 脚本 文件 
/引入 外 部 css 样式 文件 


<td colspan="4" height="30" align="center" valign="middle"><a href="del_slog_chk.php”onclick= 


"return del_chk():"> 清 除 日 志 </a></td> 





_ 国 
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</tr> 
<?php 
Sfilename = "../log.txt"; /日 志文 件 路 径 
if($f_open = fopen($filename,"r)) /是 否 以 只 读 形式 打开 文件 
{ 
while($str = fgets($f_open,255){ // 按 行 读 取 日 志文 件 内 容 
S$chr = split(",", $str); /以 “, ”为 分 隔 符 ， 返 回 数组 形式 的 记录 集 
echo "<tr>"; 
for($i = 0; $i < count($chr); $i++X{ // 循 环 输出 字段 值 
echo "<td align='center height="25'>".$chr[$i]."</td>"; } 
echo "</tr>";} 
fclose($f_open); // 关 闭 文件 
}else // 如 果 没 有 日 志文 件 时 
echo "<script>alert(' 还 没有 日 志文 件 ! ');history.go(-1);</script>"; 
Wy 
</table> 
</td></tr></table> 





对 于 无 用 的 日 志 信息 需要 定期 清除 ， 通 过 页 面 中 的 “清除 日 志 ” 超 链接 清空 日 志 ， 实 际 上 就 是 删 
除 日 志文 件 。 程 序 代码 如 下 : 


倒 程 28 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\xtgl\del_ slog_chk.php 





<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 
c_log(); /调用 自 定义 函数 


echo "<script>alert(' 删 除 成 功 ! ');location='data_stock.php';</script>"; 
?> 





3.11.4 数据 备份 的 实现 过 程 


在 病毒 肆虐 的 今天 ， 数 据 备份 的 重要 性 是 毋 需 
置疑 的 。 没 有 任何 一 家 网 站 、 一 个 系统 敢 说 自己 是 
绝对 安全 的 ， 只 要 稍 不 留神 就 会 中 招 。 本 着 对 读者 
负责 的 态度 ， 在 3.13 节 的 技术 专题 中 ， 将 详细 地 介 
绍 MySQL 数据 的 备份 与 恢复 ， 以 及 如 何在 PHP 中 
对 MySQL 进行 备份 操作 。 这 里 先 简单 了 解 一 下 本 系 
统 数据 备份 的 实现 过 程 。 数 据 备份 页 面 的 运行 结果 


办 公 自 动 化 i 
el 


PR 
如 图 3.34 所 示 。 一 
当 单 击 “ 备 份 数据 ”按钮 时 ， 系 统 执行 exec0 | 








函数 ， 该 函数 的 作用 是 执行 系统 命令 ， 通 过 系统 命 
令 来 备份 数据 。 数 据 备份 的 程序 代码 如 下 : 


全 


图 3.34 数据 备份 页 面 的 运行 结果 
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倒 程 29 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\xtgl\data_stock_chk.php 
<?php 

header("Content-type: text/html; charset=utf-8"); /设置 文 件 编码 格式 

Session_start(); 

include "../inc/chec.php"; 

include "../conn/conn.php"; 

$mysqlstr = "D:NPhpStudy\MySQLWbinWmysqldump -uroot -hlocalhost -proot --opt -B db_office > ../bak/". 
$_POST[b_name']; 

exec($mysqlstr); 

echo "<script>alert(' 备 份 成 功 ');location='data_stock.php'</script>"; 
?> 


恢复 数据 和 备份 数据 的 执行 过 程 大 同 小 异 , 都 需要 使 用 exec0 函 数 执行 系统 命令 , 只 是 命令 执行 的 
操作 是 完全 相反 的 。 恢 复数 据 的 程序 代码 如 下 : 


倒 程 30 ”代码 位 置 : 资源 包 \TM\03\oa\admin\xtgl\rebak stock_chk.php 
<?php 
header("Content-type: text/html; charset=utf-8"); // 设 置 文件 编码 格式 
session_start(); 
include "../inc/chec.php"; 
include "../conn/conn.php"; 
$mysqlstr = "D:WPhpStudy\MySQLWbinWmysql -uroot -hlocalhost -proot db_office < ../bak/".$_POST[r_name'; 
exec($mysqlstr); 
echo "<script>alert(' 恢 复 成 功 ');location='data_stock.php'</script>"; 
ea 


删除 数据 备份 实质 就 是 删除 备份 文件 ， 先 使 用 自 定义 函数 show_file0 返 回 一 个 文件 名 列表 数组 ， 
再 使 用 for 循环 语句 ， 逐 一 删除 备份 文件 。 程 序 代码 如 下 : 


倒 程 31 代码 位 置 ， 资源 包 \TM\03\oa\admin\xtgl\del_stock_chk.php 














<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../inc/func.php"; /包含 自 定义 函数 文件 
$filename = show_file(); // 调 用 自 定 义 函 数 show_file()， 返 回 文件 名 列表 


for($num = 2;$num < count($filename):$num++){ /循环 删除 文件 
unlink("../bak/".$filename[$num]); 


3 
echo "<script>alert(' 删 除 成 功 ! ');location='data_stock.php'</script>"; 
i 





3.12 开发 技巧 与 难点 分 析 


3.12.1 使 用 JavaScript 关联 多 选 列表 框 
在 本 系统 中 ， 多 个 模块 都 使 用 了 这 种 技术 ， 即 在 两 个 多 选 列表 框 之 间 产 生 关 联 ， 从 列表 框 A 中 删 
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除 的 数据 被 相应 地 添加 到 列表 框 B 中 ， 反 之 从 列表 框 B 中 删除 的 数据 会 被 添加 到 列表 框 A 中 。 

在 权限 管理 模块 中 的 用 户 组 设置 ， 就 是 用 这 种 技术 实现 的 “添加 新 用 户 组 ”功能 ， 首 先 在 左 侧 的 
多 选 列表 框 中 选取 要 添加 的 成 员 ， 单 击 “ 添 加 组 员 ” 超 链接 ， 被 选中 的 成 员 就 被 添加 到 右 侧 列表 框 ， 
同时 删除 在 左 侧 列表 框 中 的 成 员 ， 运 行 结果 如 图 3.35 所 示 。 
用 户 担 ; 防 时 组 用 户 组 : ERE 


[3 














二 Ja 页 >》 过 加 组 员 > 
‘<4 员 《人 费 际 起 员 
lr sn | mE 





图 3.35 使 用 JavaScript 关联 多 选 列表 框 


下 面 来 看 一 下 具体 的 操作 ， 首 先 创建 一 个 表单 ， 其 中 两 个 多 选 列 表 框 为 必 选 的 表单 元 素 ， 分 别 命 
名 为 left 和 right， 放 在 左 、 右 两 边 ， 其 中 left 列表 框 中 的 数据 是 从 数据 库 中 读 取 的 ，right 列表 框 则 初 
始 为 空 。 程 序 代码 如 下 : 


倒 程 32 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\qxgl\add_group.php 





<!-- 输 出 左 侧 列表 框 --> 
<SELECT name="left" size="10" multiple style="width:100px; "> 


<?php 
while($rows = mysqli_fetch_row($result)){f 
echo "<option value=".$rows[1].">".$rows[1]."</option>"; /从 数据 库 中 读 取 列 表 值 
4 
?> 
</SELECT> 
Re 
</td> 


<!-“ 添 加 ”和 “删除 ” 超 链接 一 > 
<td width="96" align="center" valign="middle"> 
<a href="#" onClick="activeList(document.form1.left,document.form1.right)"> 添 加 组 员 &gt;&gt;</a><br> 
<br> 
<a href="#" onClick="activeList(document.form1.right,document.form1.left)">&lt;&lt; 删 除 组 员 </a></td> 
<!- 右 侧 列表 框 -> 
<td colspan="2" align="center valign="middle"><select name="right" size="10" multiple style="width:100px; "> 
</select></td> 
</tr> 
<tr> 
<td height="30" colspan="4" align="center" valign="middle"> 
<input type="hidden" name="g_list" /> 
<!--“ 添 加 ”按钮 --> 
<input type="submit" value=" 添 加 " onclick="return glist()" /><input type="reset" value=" 重 置 " /> 


上 面 的 代码 主要 是 HTML 和 PHP 语言， 和 普通 的 页 面 没 有 什么 区 别 , 核心 的 语句 就 是 代码 中 的 加 


[a 
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粗 部 分 〈 不 是 注释 )， 当 单 击 “ 添 加 组 员 ” 或 “删除 组 员 ” 超 链接 时 ， 将 触发 JavaScript 脚本 事件 ， 执 
行 activeList0 函 数 。 函 数 代码 如 下 : 


倒 程 33 ”代码 位 置 ， 资源 包 \TM\03\oa\admin\js\admin jsjs 
/* 关 联 多 选 列 表 

*headStream: 源 多 选 列表 

*endStream: 目的 多 选 列表 

于 

function activeList(headStream,endStream){ 


var valueList = new Array(); /数组 ， 用 来 存储 要 移动 数据 的 value 值 
var textList = new Array(); /| 数组 ， 用 来 存储 要 移动 数据 的 text 值 
var valueTmpText = new Array(); /| 数组 ， 用 来 存储 要 移动 数据 的 option 项 
var index = 0; /数组 下 标 
”存储 源 列表 中 被 选中 的 数据 */ 
for(var i=0; i<headStream.options.length; i++}{ 
if(headStream.optionsli].selectedX{ // 判 断 元 素 是 否 被 选中 
valueList[index] = headStream.options[i].value; /存储 value 值 

textList[index] = headStream.options[i] .text; /存储 text 值 

valueTmpTextfvalueList[index]] = headStream.options[]; // 存 储 option 项 

index ++; 


» 

"向 目的 列表 中 添加 数据 ， 同 时 删除 对 应 的 源 列表 数据 */ 

for(var i=0; i<textList.length; i++){ 

var foption = document.createElement("option"); // 建 立新 的 option 项 
foption text = textList[]; 
foption.value = valueList[j]; 
endStream.add(foption); // 向 目标 列表 添加 新 建 option 项 
headStream.removeChild(valueTmpText[valueList[i]]); // 从 源 列表 中 移 除 option 项 





3.12.2 用户 组 设置 


在 后 台 的 权限 管理 模块 中 ， 放 弃 了 常用 的 “ 读 ” 和 “ 写 ”这 
样 的 常规 操作 ,而 是 采用 了 用 户 组 的 模式 。 用 户 组 的 设置 主要 是 Le en 


加 司 人 让 


两 方面 : 一 方面 是 用 户 组 成 员 , 同一 用 户 可 以 分 配 在 不 同 的 用 户 ,Lene 
组 ; 另 一 方面 是 功能 列表 , 可 以 设置 用 户 组 的 访问 权限 ,不同 的 0 
用 户 组 ， 所 看 到 的 功能 列表 也 会 不 同 。 这样， 可 以 使 用 户 的 权利 Ee 
更 加 细 化 , 防止 居心 区 测 的 用 户 恶 意 破坏 。 用 户 组 A 和 用 户 组 B 3 


的 对 比 结果 如 图 3.36 所 示 。 

在 功能 列表 的 设置 中 ， 首 先 默 认 所 有 的 功能 都 是 可 访问 的 ， 
当 有 的 功能 需要 进行 限制 时 , 就 将 其 从 用 户 组 中 删除 。 程序 代码 
如 下 : 








图 3.36 用 户 组 权限 对 比 
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倒 程 34 代码 位 置 资源 包 \TM\03\oa\admin\qxgl\pur assign.php 





<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../conn/conn.php"; // 包 含 数据 库 链接 文件 
S$sqlstr = "select id,f_name,o_group from tb_list"; /生成 SQL 语句 

?> 

<script src="..Jjs/admin_jsjs"></script> /引入 js 脚本 文件 


<link href="../css/style.css" rel="stylesheet"> /引入 外 部 css 样式 文件 


<tr><td width="75" height="20" align="center" valign="middle"> 功 能 </td> 
<td width="75" height="20" align="center" valign="middle"> 开 放 组 </td> 
<td width="75" height="20" align="center" valign="middle"> 操 作 </td></tr> 
<?php 
A* 输 出 功能 列表 项 */ 
Sresult = mysqli_query($conn, $sqlstr); 
while($rows = mysqli_fetch_row($result)X{ 
echo "<tr>"; 
for($i = 1;$i < count($rows); $i++)}{ 
echo "<td align=center valign=middle>". $rows[$i]."</td>"; 


: 
echo "<td align=center valign=middle><a href='modify_assign.php?id=".$rows[0]."> 修 改 </a></td>"; 
echo "</tr>"; 


Pesan 
?> 

</table> 

</td></tr></table> 





本 节 是 3.12.1 节 关 联 多 选 列表 框 的 延伸 ， 是 具体 应 用 关联 列表 实现 的 一 项 功能 。 在 实际 的 应 用 中 ， 
应 多 开动 脑筋 ， 才 能 用 平凡 的 技术 实现 不 平凡 的 功能 。 





3.13 ”MySQL 数据 备份 专题 


关于 备份 数据 的 重要 性 ， 这 里 就 不 再 多 做 介绍 。 本 节 先 来 了 解 常 用 数据 库 备 份 的 方式 ， 再 学 习 
MySQL 数据 备份 与 恢复 的 方法 ， 最 后 看 一 看 在 PHP 中 是 如 何 对 MySQL 数据 进行 备份 的 。 


1. 数据 库 备份 类 型 


(1) 完全 备份 。 

这 是 大 多 数 人 常用 的 方式 ， 它 可 以 备份 整个 数据 库 ， 但 需要 花费 更 多 的 时 间 和 空间 ， 所 以 一 般 推 
荐 一 周 做 一 次 。 

(2) 事务 日 志 备份 。 

事务 日 志 就 是 对 数据 库 操作 的 记录 ， 它 记录 数据 库 的 改变 ， 备 份 时 只 需要 备份 自 上 次 备份 以 来 对 
数据 库 所 做 的 改变 ， 所 以 只 需要 很 少 的 时 间 。 推 荐 每 小 时 一 次 。 
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(3) 差异 备份 。 

差异 备份 也 叫 增 量 备份 。 它 是 另 一 种 只 备份 一 小 部 分 数据 库 的 方法 ， 但 它 不 使 用 事务 日 志 ， 相 反 ， 
它 使 用 整个 数据 库 的 一 种 新 映 象 。 差 异 备份 比 最 初 的 完全 备份 小 ， 因 为 它 只 包含 自 上 次 完全 备份 以 来 
所 改变 的 数据 库 。 其 优点 是 存储 和 恢复 速度 快 。 推 荐 每 天 做 一 次 。 

(4) 文件 备份 。 

数据 库 可 以 由 硬盘 上 的 许多 文件 构成 。 如 果 这 个 数据 库 非 常 大 ， 恐 怕 一 个 晚上 也 不 能 将 它 备份 完 ， 
可 以 每 天 备份 数据 库 的 一 部 分 。 由 于 一 般 情况 下 数据 库 不 会 大 到 必须 使 用 多 个 文件 存储 ， 所 以 这 种 备 
份 不 是 很 常用 。 

2. MySQL 数据 备份 与 恢复 

MySQL 中 的 备份 是 非常 灵活 的 , 既 可 以 对 单独 的 数据 表 进 行 备份 , 也 可 以 对 整个 数据 库 进 行 备份 ， 
下 面 分 别 进行 介绍 。 

(1) 数据 表 备 份 。 

备份 数据 表 的 语法 如 下 : 


SELECT * INTO OUTFILE ‘file_name' FROM tbl_name 
























































回 fle name: 备份 的 文件 名 ， 如 user.sql。 

回 tbl_ name: 备份 的 数据 表 名 ， 如 tb_users。 

使 用 该 备份 语句 首先 要 登录 到 MySQL， 并 进入 到 相对 应 的 数据 库 中 ， 如 要 备份 的 表 tb_users 是 属 
于 数据 库 db_office 的 ， 备 份 文件 名 为 users.sql， 则 完整 的 备份 流程 如 图 3.37 所 示 。 


| 昌 jx 
癌 








图 3.37 数据 表 备份 





备份 后 的 文件 保存 在 数据 表 所 在 目录 中 ， 如 本 例 中 表 tb_users 所 在 的 目录 为 D:\AppServIMySQI\ 
data\db_office， 则 备份 的 sql 文件 就 保存 在 这 个 路 径 下 。 
如 果 要 恢复 备份 文件 ， 代 码 如 下 : 


LOAD DATA INFILE "usersql' REPLACE into table 'tb_users’ 
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(2) 数据 库 备 份 。 
数据 库 备 份 时 ， 可 以 不 用 登录 到 SQL 中 ， 直 接 使 用 mysqldump 命令 即 可 。 该 命令 格式 如 下 : 
mysqldump -uroot -proot db_office > c:\\remark .txt 


备份 的 数据 表 ， 实 际 就 是 一 个 完整 的 建 库 流程 ， 如 果 要 恢复 备份 ， 直 接 用 MySQL 命令 即 可 。 语 
旬 如 下 : 


mysql -uroot -proot db_office < c:\remark .txt 


mysqldump 语句 的 功能 十 分 强大 ， 备 份 只 是 其 中 的 一 个 功能 ， 感 兴趣 的 读者 可 以 参考 mysqldump 
的 帮助 文档 。 


3. 在 PHP 中 备份 数据 库 


在 PHP 中 备份 数据 库 ， 主 要 通过 exec0 函 数 执行 系统 命令 。 
exec0 函 数 的 格式 如 下 : 


string exec(string command) 
参数 command 为 要 执行 的 系统 命令 。 这 里 为 备份 数据 表 的 命令 ， 代 码 如 下 : 


exec("D:\PhpStudy\MySQL\\bin\Imysqldump -uroot -hlocalhost -proot --opt -B db_office > ../bak/remark.sql"); 














\ po 
命令 ， 则 必须 给 出 命令 的 完整 路 径 ， 否则， 就 会 出 现 文件 大 小 为 0KB 的 结果 . 


本 章 通过 一 个 完整 的 办 公 自 动 化 管理 系统 开发 过 程 ， 向 读者 讲述 了 系统 开发 的 几 个 重要 环节 ， 还 
包括 几 个 主要 模块 的 功能 实现 ， 最 后 又 着 重 介绍 了 MySQL 备份 的 相关 知识 。 通 过 本 章 的 学 习 ， 读 者 
对 如 何 开发 一 个 真实 的 项 目 会 有 进一步 的 了 解 。 相 信 读 者 通过 自己 动手 实践 ， 完 成 一 个 功能 相似 的 办 
公 自 动 化 管理 系统 是 完全 不 成 问题 的 。 


铭 成 在 线 考 试 系统 


(PHP+MYySQL+ 组 件 化 开发 实现 ) 


随 着 科技 的 发 展 ， 网 络 技术 已 经 深入 到 人 们 的 日 常生 活 中 ， 同 时 
带 来 了 教育 方式 的 一 次 变革 。 而 网 络 考试 则 是 一 个 很 重要 的 方向 。 基 
于 Web 技术 的 网 络 考试 系统 可 以 借助 于 遍布 全 球 的 Internet 进行 。 因 
此 考试 既 可 以 在 本 地 进行 ， 也 可 以 在 异地 进行 ， 大 大 拓展 了 考试 的 灵 
活性 ， 并 且 缩 短 了 传统 考试 要 求 老师 打印 试卷 、 安排 考试 、 监 考 、 收 
集 试卷 、 评 改 试卷 、 讲 评 试卷 和 分 析 试 卷 这 个 漫长 而 复杂 的 过 程 ， 使 
考试 更 趋 于 客观 、 公 正 。 本 章 介 绍 了 一 个 具有 在 线 考 试 、 即 时 阅卷 、 
成 绩 查 询 以 及 试卷 和 考题 信息 管理 等 功能 的 铭 成 在 线 考试 系统 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

”掌握 铭 成 在 线 考试 的 开发 过 程 

Wm 掌握 组 件 化 天 发 

H 理解 MVC 模式 

MH 党 所 数据 库 对 象 关系 映射 ORM i 

Wm 掌握 Ajax 技术 加 
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41 开发 背景 











随 着 教育 改革 的 不 断 深入 ， 近 几 年 来 出 现 了 一 些 新 的 教育 形式 ， 如 网 上 授课 、 网 上 考试 等 。X xX x 
高 校 是 一 所 师资 力量 非常 雄厚 的 学 校 ， 随 着 院 校 的 扩招 ， 学 生 数 量 不 断 增加 。 为 了 适应 新 形式 的 发 展 ， 
改变 传统 的 教学 模式 ， 方 便 学 生 随 时 随地 地 对 自己 的 学 习 情 况 进行 检测 ， 减 轻 教师 的 工作 压力 ， 现 委 
托 其 他 公司 开发 网 上 在 线 考试 系统 。 这 种 无 纸 的 网 络 考试 系统 ， 使 考 务 管理 突破 时 空 限制 ， 提 高 考试 
工作 效率 和 标准 化 水 平 ， 使 学 校 管 理 者 、 教 师 和 学 生 可 以 在 任何 时 候 、 任 何 地 点 通过 网 络 进行 考试 。 
网 络 在 线 考试 系统 已 经 成 为 教育 技术 发 展 与 研究 的 方向 。 


42 系统 分 析 


4.2.1 需求 分 析 


随 着 计算 机 技术 的 发 展 和 网 络 技术 的 日 益 成 熟 ， 通 过 网 络 进行 信息 交流 已 成 为 一 种 快捷 的 交互 方 
式 。 在 这 种 网 络 环境 下 ， 学 校 或 考试 机 构 希 望 通过 建立 网 络 在 线 考试 网 站 来 扩大 知名 度 、 降 低 管 理 成 
本 和 减少 人 力 、 物 力 的 投资 ， 从 而 为 考生 提供 更 全 面 、 更 灵活 的 服务 ， 并 全 面 、 准 确 地 对 考试 进行 跟 
踪 和 评价 。 与 此 同时 ， 考 生 希 望 根据 自己 的 学 习 情况 进行 测试 ， 并 能 够 得 到 客观 、 科 学 的 评价 ， 教 务 
人 员 希 望 能 够 有 效 地 改进 现 有 的 考试 模式 ， 提 高 考试 效率 。 

通过 实际 情况 的 调查 ， 要 求 网 络 在 线 考试 系统 具有 以 下 功能 : 
界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵 活 ， 树 立 企业 形象 。 

要 求实 现在 线 考试 功能 ， 自 动 核算 考试 成 绩 。 

要 求 提供 考试 时 间 倒 计时 功能 ， 使 考生 实时 了 解 考试 剩余 时 间 。 
要 求 系统 自动 阅卷 ， 保 证 考试 成 绩 真实 有 效 。 

要 求 考生 可 以 查询 考试 成 绩 ， 以 保证 信息 安全 。 

系统 运行 稳定 、 安 全 可 靠 。 

要 求 对 考生 及 考题 信息 进行 严格 管理 。 


4.2.2 ”可行 性 分 析 


可 行 性 分 析 的 目的 就 是 要 用 最 小 的 代价 在 尽 可 能 短 的 时 间 内 确定 问题 是 否 能 够 解决 。 通 过 分 析 解 
法 的 利弊 ， 来 判定 系统 目标 和 规模 是 否 现实 ， 系 统 完成 后 所 能 带 来 的 效益 是 否 达 到 值得 去 投资 开发 这 
个 系统 的 程度 。 网 络 在 线 考 试 系统 的 可 行 性 可 从 以 下 两 方面 考虑 。 

1. 经 济 可 行 性 


定期 地 组 织 考试 是 各 个 院 校 及 时 掌握 学 生 学 习 成 绩 的 有 效 方式 。 利 用 网 络 在 线 考试 系统 ， 一 方面 
可 以 节省 人 力 资源 ， 降 低 考 试 成 本 ; 另 一 方面 ， 在 线 考试 系统 能 够 快速 进行 考试 和 评分 ， 体 现 出 考试 
的 客观 与 公正 性 。 


全 


办 区 办 办 办 旬 凶 
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2. 技术 可 行 性 


开发 一 个 网 络 在 线 考试 系统 ， 涉 及 的 最 核心 的 技术 问题 就 是 如 何 实现 在 不 刷新 页 面 的 情况 下 实时 
显示 考试 时 间 及 剩余 时 间 ， 并 做 到 到 达 考 试 结束 时 间 时 自动 提交 试卷 的 功能 。 通 过 JavaScript 技术 可 以 
轻松 实现 这 些 功 能 ， 这 为 网 络 在 线 考试 系统 的 开发 提供 了 技术 保障 。 


43 系统 设计 


4.3.1 系统 目标 


根据 前 面 所 作 的 需求 分 析 及 用 户 的 需求 可 知 ， 铭 成 在 线 考试 系统 属于 中 小 型 软件 ， 在 系统 实施 后 
应 达到 以 下 目标 
采用 开放 、 动 态 的 系统 架构 ， 加 强 用 户 与 网 站 的 动态 交互 性 。 
具有 空间 性 。 被 授权 的 用 户 可 以 在 异地 登录 网 络 在 线 考试 系统 ， 无 须 到 指定 地 点 进行 考试 。 
操作 简单 方便 、 界 面 简洁 美观 。 
系统 提供 考试 时 间 倒计时 功能 ， 使 考生 实时 了 解 考试 剩余 时 间 。 
实现 自动 提交 试卷 的 功能 。 当 考试 时 间 到 达 规定 时 间 时 ， 如 果 考 生还 未 提交 试卷 ， 系 统 将 自 
动 交卷 ， 以 保证 考试 严肃 、 公 正 地 进行 。 
回 系统 自动 阅卷 ， 保 证 成 绩 真 实 准确 。 
回 ”考生 可 以 查询 考试 成 绩 。 
回 ”系统 运行 稳定 、 安 全 可 靠 。 


国共 办 加 


4.3.2 ”系统 功能 结构 


铭 成 在 线 考 试 系统 中 的 各 个 功能 模块 均 围绕 着 前 台数 据 展示 、 后 台 逻 辑 控制 这 两 个 方面 进行 设计 。 
铭 成 在 线 考试 系统 功能 结构 图 如 图 4.1 所 示 。 

















铭 城 考试 系统 
系统 功能 系统 维护 
上 用 | | 试 | | 试 | | 考 科 | | 试 | | 考 
| 户 | | 户 || 卷 | | 卷 | | 医 县 | | 时 | 齐 
相 | 登 | | 展 | | 评 | | 计 管 | | 管 | | 管 
Hil 分 | | 时 理 | | 理 | | 理 
图 4.1 功能 结构 图 
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4.3.3 系统 流程 图 


铭 成 在 线 考试 系统 的 业务 流程 : 首先 系统 维护 人 员 编辑 一 份 试卷 、 添 加 试卷 、 添 加 问题 、 添 加 答 
案 、 单 选 /多 选 、 总 分 等 ; 然后 用 户 选择 试卷 、 开 始 答题 并 计时 、 回 答 试题 、 提 交 试 卷 等 ， 最 后 由 后 
台 判 断 从 前 台 传递 给 后 台 的 答案 对 错 、 试 卷 评分 、 完 成 评 卷 等 。 铭 成 在 线 考试 系统 考试 流程 如 图 4.2 
所 示 。 





昌 








登录 系统 是 否 到 时 
否 
2 2 
选择 科目 继续 考试 是 
vy vy 
| 选择 试卷 提交 试卷 。 
否 
| . 
| 是 否 有 考题 
是 
+ La 
开始 考试 。 | 结束 


图 4.2 系统 流程 图 
4.3.4 系统 预览 


铭 成 在 线 考 试 系统 由 多 个 页 面 组 成 ， 下 面 仅 列 出 几 个 典型 页 面 ， 其 他 页 面 参见 资源 包 中 的 源 程序 。 

系统 首页 如 图 4.3 所 示 。 用 户 参 数 考试 ， 首 先 需要 登录 ， 考 试 登录 页 面 运行 效果 如 图 4.4 所 示 。 如 
果 用 户 没 有 账号 ， 则 需要 注册 账号 ， 考 试 注册 页 面 运行 效果 如 图 4.5 所 示 。 登 录 成 功 后 ， 用 户 选 择 考试 
科目 页 面 如 图 4.6 所 示 ， 该 页 面 列举 出 所 有 考试 科目 。 选 择 考试 试卷 页 面 如 图 4.7 所 示 ， 该 页 面 用 于 选 
择 所 有 科目 下 的 试卷。 在 线 答题 页 面 如 图 4.8 所 示 ， 该 页 面 实现 在 线 答题 功能 ， 同 时 提供 了 显示 考试 剩 
余 时 间 及 自动 提交 试卷 功能 。 考 试 成 绩 页 面 如 图 4.9 所 示 ， 该 页 面 显 示 了 本 次 考试 的 成 绩 。 

网 站 管理 员 可 以 在 后 台 设置 考试 科目 、 试 卷 和 考题 等 相关 内 容 ， 后 台 登 录 页 面 如 图 4.10 所 示 ， 后 
台 考 试 科目 设置 页 面 如 图 4.11 所 示 ， 填 写 试题 页 面 如 图 4.12 所 示 。 
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大 只 一 库 法 短 ， 随 出所 闪 有 


O = 1 











用 站 知 如、 实时 提要 














图 4.3 系统 首页 
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图 4.4 登录 页 面 





图 4.5 注册 页 面 


选择 考试 问卷 





7 4 7 bE 


4.6 选择 考试 科目 页 面 效果 
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ooln 


只 长 春 代数 试题 (着 一 Ee 





尊敬 的 "mr' : 您 于 2017-12-18 18:30:30 参与 【小 学 5 年 级 期 中 考试 数学 题 】 考 试 ， 
用 时 0 分 10 秒 ， 考 试 成 绩 为 : 0 分 


EE 





图 4.9 本 次 考试 成 绩 页 面 效 果 
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铭 成 在 线 考试 系统 后 台 





图 4.10 后 台 登 录 页 面 








网 PR: 








图 4.12 填写 试题 页 面 效果 
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4.3.5 ”开发 环境 


在 开发 网 络 在 线 考试 系统 时 , 该 项 目 使 用 的 
软件 开发 环境 如 下 。 


回 


回回 加 罗 加 


4.3.6 


操作 系统 : Windows 07 及 以 上 /Linux 
(推荐 )。 

集成 开发 环境 : phpStudy2016。 

PHP 软件 : PHP 7.0。 

开发 工具 : PhpStorm 9。 

包 管理 工具 : Composer。 

浏览 器 : 谷歌 浏览 器 。 


文件 夹 组 织 结构 


铭 成 在 线 考试 系统 的 项 目 目录 结构 如 图 4.13 


AN 


4.4.1 





数据 库 分 析 


v 





Exam [MFFC] (Di\php 


TY 户 app 一 一 一 一 一 一 一 一 一 应 用 目录 
》 站 controllers 一 一 一 一 一 控制 器 目录 
》 站 models 一 一 一 一 一 一 一 模型 目录 
》 门 views 一 一 一 一 一 一 一 视图 目录 

* 口 confg 配置 文件 

» Ddata 数据 库 文件 

》 站 public 一 一 一 一 一 一 一 一 资源 文件 

》 站 sevices 一 一 一 一 一 一 一 服务 目录 

» Dvendor 组 件 目录 
R hacces Apache 配 置 文件 
足 bootstrap.php 一 一 一 一 一 一 启动 文件 
冰 composerjson Composer 配 置 文件 
目 composerlock Composer 锁 文件 
目 README.md 一 一 一 一 一 一 使 用 说 明文 件 


图 4.13 铭 成 在 线 考试 系统 的 文件 夹 组 织 结构 


4.4 数据 库 设计 


本 系统 采用 MySQL 作为 数据 库 ， 数 据 库 名 称 为 exam， 其 数据 表 名 称 及 作用 如 表 4.1 所 示 。 
表 4.1 数据 库 表 结构 




















表 名 作 用 

admin 用 于 存储 管理 员 用 户 信息 

Users 用 于 存储 用 户 信 息 

subjects 用 于 存储 考试 科目 信息 

papers 用 于 存储 试卷 信息 

questions 用 于 存储 考题 信息 
4.4.2 ”数据 表 结 构 


由 于 篇 幅 所 限 ， 这 里 只 给 出 较 重要 的 数据 表 的 部 分 字段 ， 完 整数 据 表 请 参见 本 书 附带 资源 包 。 


回 


subjects (科目 表 ) 


表 subjects 用 于 保存 考试 科目 数据 信息 ， 其 结构 如 表 4.2 所 示 。 
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表 4.2 考试 科目 表 结构 




















字 段 名 数据 类 型 注 
id | ints) 主键 

subject_ name | varchar(255) 科目 名 称 
updated at | datetime 更 新 时 间 













created at datetime 创建 时 间 





回 papers〈 试 卷 表 ) 
表 papers 用 于 存储 试卷 信息 ， 其 结构 如 表 4.3 所 示 。 


表 4.3 试卷 表 结 构 


默 认 值 允许 为 空 自动 递增 备注 
i NO 是 主键 
paper_name varchar(255) YES 试卷 名 称 


teacher name | vaaos5 | | 出 题 老 师 名 称 
sbjectid | | | ys | 所 属 科目 ID 
subject mame | varchac5) | | yes | 所 属 科目 名 称 
eeaedat | ume | | ves | 创建 时 间 
wiedat | dem | | yes | 修改 时 间 


total_time 答题 时 间 ， 单 位 是 秒 














回 “questions (考题 表 ) 
表 questions 用 于 存储 考题 信息 ， 其 结构 如 表 4.4 所 示 。 




















表 4.4 考题 表 结 构 

字 段 名 数 据 类 型 默 认 值 自动 递增 备 注 
id int(8) 是 主键 

paper id int(8) 试卷 ID 
paper_name varchar(255) 试卷 名 称 

order int(8) 题 号 

score int(8) 分 数 

title varchar(255) 题目 

type tinyint(1) 类 型 ，1 单 选 ，2 多 选 
‘options text 选项 

answers varchar(255) 答案 














results (考试 成 绩 表 ) 
表 results 用 于 存储 考试 成 绩 信息 ， 其 结构 如 表 4.5 所 示 。 
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表 4.5 考试 成 绩 结构 






自动 递增 
id int(8) 主键 
user id int(8) 用 户 ID 
Usemame varchar(255) 考生 姓名 
paper id int(8) 试卷 ID 
SCore int(8) 得 分 
start_time datetime 开始 时 间 
end time datetime 结束 时 间 
paper name varchar(255) 试卷 名 称 








































4.4.3 数据 表 关 系 


铭 成 在 线 考试 系统 中 数据 表 的 关系 为 : 一 个 考试 科目 〈subjects 表 ) 对 应 多 份 试卷 (papers 表 ) ， 一 
份 试卷 对 应 多 道 试 题 questions 表 ) 。 一 份 试卷 对 应 多 个 成 绩 (result) 。 数 据 库 关系 图 如 图 4.14 所 示 。 





paper_name aper_name 
加 teacher_name Re 
Subject_name Be subjectid i 
updated_at subjectname te 
crealed_ 引 created_at type 
updated_at opin 


total_time 











图 4.14 数据 库 关系 图 


45 组 件 开 发 





4.5.1 什么 是 组 件 


组 件 是 一 组 打包 的 代码 ， 是 一 系列 相关 的 类 、 接 口 和 Trait， 用 于 帮助 我 们 解决 PHP 应 用 中 某 个 具 
体 问 题 。 例 如 ， 你 的 PHP 应 用 需要 收发 HTTP 请 求 ， 可 以 使 用 现成 的 组 件 如 guzzle/guzzle 实现 。 我 们 
使 用 组 件 不 是 为 了 重新 实现 已 经 实现 的 功能 ， 而 是 把 更 多 时 间 花 在 实现 项 目的 长 远 目标 上 。 
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4.5.2 为 什么 使 用 组 件 


很 多 开发 者 用 了 大 量 时 间 创 建 、 优 化 和 测试 专门 的 组 件 ， 我 们 使 用 已 有 的 组 件 开 发 ， 就 可 以 避免 
重复 造 轮子 。 一 个 优秀 的 PHP 组 件 具 备 以 下 特性 : 

回 ”作用 单一 : 专注 于 解决 一 个 问题 ， 而 且 使 用 简单 的 接口 封装 功能 。 

加 ”小 型 : 小 巧 玲珑 ， 只 包含 解决 某 个 问题 所 需 的 最 少 代码 。 

回 合作 : PHP 组 件 之 间 可 以 良好 合作 ， 组 合 在 一 起 实现 大 型 项 目 。 

回 ”测试 良好 : 本 身 提供 测试 ， 而 且 有 充足 的 测试 覆盖 度 。 

加 ”文档 完善 :应 该 提供 完善 的 文档 ， 能 让 开发 者 轻易 安装 、 理 解 和 使 用 。 


4.5.3 查找 组 件 


在 Packagist (官方 网 址 : https://packagist.org) 中 可 以 查找 各 种 PHP 组 件 ， 这 个 网 站 用 于 收集 PHP 
组 件 ， 最 好 的 PHP 组 件 在 Packagist 中 都 能 找到 。Packagist 网 站 如 图 4.15 所 示 。 比 如 我 们 想 使 用 一 个 
http 组 件 用 于 收发 HTTP 消息 ， 在 搜索 框 中 搜索 “http”， 会 查找 到 所 有 关于 HTTP 的 组 件 。 


Packagist ene rocog 








Getting Started Publishing Packages 
Define Your Dependencies Define Your Package 





] 


For more Informatlon about packages verslons usage see the composel 
documentation. 





Install Composer In Your Project 


Runthis in your command Iine: 
Commit The File 


Add the compeser json to yoUr BI Or other VCS repostory and Commit It 





Publish It 


图 4.15 Packagist 首页 
4.5.4 使 用 PHP 组 件 


使 用 PHP 组 件 必须 解决 两 个 问题 : 依赖 管理 和 自动 加 载 。 可 以 使 用 Composer 去 解决 它 。Composer 
是 安装 PHP 组 件 的 工具 ， 也 是 PHP 组 件 的 依赖 管理 器 ， 运 行 在 命令 行 中 。 

Composer 能 和 Packagist 配合 ， 如 果 需 要 通过 Composer 下 载 组 件 ，Composer 会 通过 Packagist 获 
取 相 关 组 件 。 
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7/ 
I 
度 很 慢 ， 请 使 用 Packagist/Composer 中 国 全 量 镜 像 安装 PHP 组 件 。 


在 项 目 根 目录 下 创建 composrjson 文件 ， 安 装 组 件 的 关键 代码 如 下 : 
{ 


"require": { 
"noahbuscher/macaw": "dev-master", 
"illuminate/database": ™*", 
"jasongrimes/paginator": "~1.0" 
» 
"repositories": { 
"packagist": { 
"type": "composer", 
"url": "https://packagist.phpcomposer.com" 
} 
} 


composr.json 文件 中 引入 了 3 个 组 件 ， 下 面 简 要 介绍 一 下 每 个 组 件 。 
1. noahbuscher/macaw 路 由 组 件 


noahbuscher/macaw 是 一 个 简单 的 路 由 组 件 ， 特 点 是 体积 小 ， 反 应 快 ， 使 用 方便 。 组 件 地 址 : 
https://packagist.org/packages/noahbuscher/macaw。 使 用 该 路 由 组 件 的 步骤 如 下 : 
(1) 导入 类 文件 。 代 码 如 下 : 


use\NoahBuscher\Macaw\Macaw; 











(2) 配置 路 由 文件 。 路 由 文件 的 目录 是 “/config/routes.php”， 该 文件 中 ， 配 置 了 铭 成 在 线 网 站 的 
路 由 信息 。 路 由 中 常用 的 方式 有 3 种 : 普通 GET 方式 、 带 参数 的 GET 方式 和 POST 方式 。 示 例 代码 
如 下 : 

Macaw::get('/subjects', 'HomeController@subjects’); 


Macaw::get('/subject/(:num)', "HomeController@papers'); 
Macaw::post('/result', 'HomeController@result’); 


当 访 问 “/subjects” 时 ， 执 行 app\controllers\HomeController.php 文件 下 HomeController 类 的 subjects0 
方法 ; 当 访 问 “/subject/1” 时 , 执行 HomeController 类 的 papers0 方 法 ; 当 以 post 方式 提交 表单 到 “/result” 
路 径 时 ， 执 行 HomeController 类 的 resutl(0 方 法 。 

(3) 调度 路 由 。Macaw::get0 方 法 只 是 获取 路 由 配置 信息 ， 而 真正 执行 路 由 调度 的 是 Macaw:: 
dispatch0 方 法 。 所 以 ， 通 常 在 路 由 配置 文件 的 最 后 ， 添 加 如 下 代码 : 


Macaw::dispatch(); 
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2. illuminate/database 数据 库 组 件 


illuminate/database 是 一 个 完整 的 数据 库 工具 包 , 提供 了 漂亮 、 简 洁 的 ActiveRecord 实现 来 和 数据 
库 进行 交互 。 每 个 数据 库 表 都 有 一 个 对 应 的 “模型 ”可 用 来 跟 数 据 表 进 行 交互 。 可 以 通过 模型 查询 数据 
表 内 的 数据 ， 以 及 将 记录 添加 到 数据 表 中 。 组 件 地 址 为 https://packagist.org/packages/illuminate/database。 

使 用 该 数据 库 组 件 的 步骤 如 下 : 

(1) 创建 Capsule 管理 器 实例 。 代 码 如 下 : 


use llluminate\Database\Capsule\Manager as Capsule; 
$capsule = new Capsule; /实例 化 类 


(2) 添加 配置 信息 ， 代 码 如 下 : 
$capsule->addConnection(require BASE_PATH.'/config/database.php’); 


上 述 代码 中 调用 了 addConnection0 方 法 ， 并且 传递 数据 库 的 配置 参数 。config/database.php 代码 如 下 : 





<?php 

return [ 
'driver => ,mysql'， /| 数据 库 类 型 
host => 127.0.0.1, /主机 地 址 
"database' => 'exam'， /| 数据 库 名 
username' => 'root’, /| 数据 库 用 户 名 
"password' => 'root', /| 数据 库 密码 
'charset => 'utf8', /| 数据 库 编码 
"collation' => 'utf8_general_ci', /整理 


'prefix =>" // 表 前 缀 
上 


(3) 调用 bootEloquent() 方 法 启动 Eloquent ORM。 代 码 如 下 : 





$capsule->bootEloquent(); 
(4) 使 用 Eloquent ORM。 示 例 代码 如 下 : 


class User extends llluminate\Database\Eloquent\Model 0 
$users = User::where('votes', '>", 1)->get(); 
3. jasongrimes/paginator 分 页 组 件 


jasongrimes/paginator 是 一 个 轻 量 级 的 分 页 组 件 ， 在 分 页 列表 中 ， 可 以 使 用 “前 一 页 ”和 “后 一 页 ” 
等 样式 。 组 件 地 址 为 https://packagist.org/packages/jasongrimes/paginator。 
使 用 该 分 页 组 件 的 步骤 如 下 : 
(1) 导入 类 文件 。 代 码 如 下 : 


use JasonGrimes\Paginator; 


(2) 实例 化 分 页 类 。 实 例 化 分 页 类 需要 传递 4 个 参数 ， 代 码 如 下 : 
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Stotalltems = 1000; /总 条 数 
$itemsPerPage = 50; /每 页 显示 数量 
$currentPage = 8; /当前 所 在 页 码 
$urlPattern = "/foo/page/(:-num)'; JWurl 类 型 


S$paginator = new Paginator($totalltems, $itemsPerPage, $currentPage, $urlPattern); // 实 例 化 分 页 类 


(3) 模板 赋值 。 将 生成 的 Spaginator 变量 赋值 到 模板 页 面 。 示 例 代码 如 下 : 


<html> 
<head> 
<I- The default, built-in template supports the Twitter Bootstrap pagination styles. --> 
<link rel="stylesheet" href="//maxcdn.bootstrapcdn.com/bootstrap/3.2.0/css/bootstrap.min.css"> 
</head> 
<body> 
<?php 
echo $paginator; /分 页 内 容 
?> 
</body> 
</html> 








4.5.5 入口 文件 


配置 网 站 时 ， 需 要 将 Exam/public 文件 夹 放置 在 网 站 根 目 录 ， 并 且 设 置 .htaccess 文件 , 将 index.php 
文件 作为 项 目的 入 口 文件 。 设 置 完成 后 , 访问 网 站 任何 路 由 , 都 会 从 index.php 入 口 文件 开始 执行 程序 ， 
index.php 代码 如 下 : 





<?php 


/定义 PUBLIC_PATH 
define('PUBLIC_PATH', _DIR_); 


// 启 动 器 
@ require PUBLIC_PATH.'/../bootstrap.php'; 


// 路 由 配置 、 开 始 处 理 
@ require BASE_PATH.'/config/routes.php'; 
< 代码 贴 二 
@ bootstrap.php: 项 目 启动 文件 。 
@ routes.php: 路 由 文件 。 


index.php 入 口 文件 中 引入 了 bootstrap .php 项 目 启动 文件 ， 该 文件 主要 为 项 目 执行 前 做 准备 ， 代 码 
如 下 : 


<?php 
/定义 BASE_PATH 


@® 
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define(BASE_PATH'，_DIR_): 

// 导 入 llluminate\Database 

@ use llluminate\Database\Capsule\Manager as Capsule; 

//Autoload 自动 载 入 

四 require BASE_PATH.vendorautoload.php'; 

/Eloquent ORM 

$capsule = new Capsule; 

$capsule->addConnection(require BASE_PATH./config/database.php'); 
$capsule->bootEloquent(); 


© date_default_timezone_set('Asia/Shanghai"); /设置 时 区 





< 代码 贴 十 
@ use: 导入 数据 库 组 件 。 
@ autoload.php: 自动 加 载 类 文件 。 
@ date default_ timezone setO: 该 函数 用 于 设置 时 区 。 默 认 时 间 是 格林 威 治标 准时 间 ， 和 北京 时 间 差 了 8 个 小 时 。 


4.6 考试 科目 模块 设计 


4.6.1 考试 科目 模块 概述 


考试 科目 模块 主要 包括 获取 并 显示 考试 科目 和 获取 并 显示 指定 考试 科目 的 所 有 试卷 2 个 主要 功能 
页 面 。 操 作 流程 是 “选择 科目 ”/“ 选 择 试卷 ” /“ 开 始 考试 ”。 用 户 单 击 “ 在 线 考试 ”导航 ， 进 入 到 考 
试 科目 页 面 ， 该 页 面 显示 所 有 考试 的 科目 ， 如 “语文 ”“ 数 学 ”“ 英 语 ” 等 。 当 用 户 选择 完 科 目 后 ， 
进入 该 科目 的 试卷 选择 页 面 。 用 户 需 要 选择 一 门 试卷 ， 如 “小 学 5 年 级 期 中 考试 考题 ”， 如 果 该 试卷 
下 存在 考题 ， 则 开始 考试 ， 否 则 提示 “该 试卷 下 暂 无 考题 ， 请 选择 其 他 试卷 ”。 


4.6.2 ”考试 科目 模块 技术 分 析 


在 铭 成 在 线 考试 系统 中 ， 通 过 查询 subjects 表 ， 获 得 考试 科目 的 数据 。 根 据 subjects 表 中 的 ID, 查 
询 paper 表 中 所 有 subject id 为 该 ID 的 数据 , 即 为 该 科目 下 的 所 有 试卷 信息 。 然 后 再 根据 paper 表 的 人 ， 
查询 所 有 questions 表 中 paper id 为 该 的 数据 ， 即 为 该 试卷 的 考题 信息 。 


4.6.3 获取 并 显示 考试 科目 


国 获取 并 显示 考试 科目 使 用 的 数据 表 : subjects 
1. 获取 考试 科目 数据 


考试 科目 数据 存储 在 subjects 表 中 ， 我 们 使 用 illuminate/database 数据 库 组 件 ， 通 过 表 名 ::all0 方 法 
来 获取 subjects 表 全 部 数据 ， 关 键 代码 如 下 : 
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倒 程 01 代码 位 置 资源 包 \TM\04\Exam\app\controllers\HomeController.php 


/选择 科目 
public function subjects() 





$subjects = Subjects::all(); // 获 取 所 有 考试 科目 数据 
Sthis->view = View::make('home.subjects')->with('subjects',$subjects);  / 泻 染 视图 模板 
} 


上 述 代码 中 ，$subjects 是 一 个 存储 考试 科目 信息 的 集合 ， 我 们 使 用 noahbuscher/macaw 路 由 组 件 ， 
通过 View::make0 方 法 生成 路 由 ， 并 将 $subjects 变量 传递 到 模板 文件 。 


2. 显示 考试 科目 


前 台所 有 模板 文件 均 在 app\views\home 目录 下 ， 其 中 头 部 模板 、 导 航模 板 和 底部 信息 模板 都 是 公 
共 模 板 ， 我 们 将 其 放置 在 partials 目录 下 。 具 体 代码 如 下 : 


倒 程 02 ”代码 位 置 ， 资源 包 \TM\04\Exam\app\views\home\subjects.php 


<!- 引 入 头 部 文件 -> 
@ <?php require(BASE_PATH.'/lapp/views/partials/head.php');?> 
<!- 引 入 导航 文件 -> 
<?php require(BASE_PATH.'/lapp/views/partials/home_nav.php');?> 
<div class="container" style="padding-top: 65px;"> 

<img alt="" src="./images/fenlei01.png" class="img-responsive"> 
</div> 




















<div class="container" style="padding-top: 50px;"> 
<img alt=" src="./images/fenlei02.png" class="img-responsive"> 
</div> 
<div class="container subjects"> 
<table class="table table-bordered " style="border-spacing: 30px;margin-bottom: 100px" > 
<tr> 
< 上 -遍历 数据 -> 
© <?php foreach($subjects as $subject) :?> 
<td> 
<div style="height: 80px;padding-top: 25px;"> 
<span class="glyphicon glyphicon-pencil"></span> 
&nbsp;&nbsp;&nbsp; 
© <a href="/subject/<?php echo $subject->id?>"> 
<strong class="text-danger" style="font-size: 18px;"> 
<?php echo $subject->subject_name?></strong> 
</a> 
</div> 
</td> 
<?php endforeach?> 
</tr> 
</table> 
</div> 
<!-- 引 入 底部 文件 --> 
<?php require(BASE_PATH.'/app/views/partials/home_footer.php');?> 
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<script type="text/javascript"> 
function hideFooter(){ 
$("#footer").hide(); 
} 
</script> 
</body> 
</html> 





4 代码 贴 二 
@ require: 引入 公共 文件 。 
@ foreach: foreach 遍历 数据 ， 注 意 foreach 和 endforeach 是 一 对 标签 。 
@ $subject->id: 可 以 使 用 箭头 符号 从 集合 中 获取 数据 。 


4.6.4 ”获取 并 显示 指定 考试 科目 的 所 有 试卷 


国 ”获取 并 显示 指定 考试 科目 的 所 有 试卷 使 用 的 数据 表 : papers 
1. 获取 试卷 数据 


由 于 每 个 考试 科目 中 可 能 存在 多 张 考试 试卷 ， 为 了 获取 指定 考试 科目 下 的 全 部 考试 试卷 ， 将 通过 
某 一 考试 科目 的 ID 获取 该 考试 科目 下 的 全 部 考试 试卷 。 也 就 是 说 ,考试 科目 与 该 考试 科目 下 的 全 部 考 
试 试卷 存在 一 对 多 的 关系 。 关 键 代 码 如 下 : 


倒 程 03 ”代码 位 置 ; 资源 包 \TM\04\Exam\app\controllers\HomeController.php 





public function papers($subject_id) 


{ 

$papers = Papers::where('subject_id', $subject_id)->orderBy('created_at','desc')->get(); // 获 取 试 卷 信息 ， 并 
根据 创建 时 间 排 序 

// 泻 染 视图 模板 

Sthis->view = View::make('home.papers')->with('subject_id', $subject_id) 

->with('papers',$papers); 
二 一 
上 述 代 码 中 , 由 于 添加 了 subject_id 这 个 筛选 条 件 , 所 以 使 用 get0 来 获取 数据 , 并且 使 用 orderByO 
方法 对 creaded_at 字段 降序 排序 。 


2. 显示 试卷 信息 


当 获 取 到 某 一 个 科目 下 的 所 有 试卷 后 ， 需 要 遍历 所 有 试卷 信息 ， 并 且 以 下 拉 列 表 的 方式 展示 试卷 
信息 。 当 选择 完 试卷 后 ， 单 击 “ 开 始 考试 ”按钮 ， 使 用 Ajax 异步 提交 的 方式 检查 试卷 下 是 否 有 考题 ， 
如 果 有 ， 则 跳 转 至 考试 页 面 ， 否 则 提示 错误 信息 。 显 示 试 卷 的 代码 如 下 : 


倒 程 04 代码 位 置 资源 包 \TM\04\Exam\app\views\home\papers.php 





<div class="input-group" style="padding-top: 50px;"> 
<span class="input-group-addon"> 
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</span> 
<select id='papers' name="papers" class="form-control"> 
© <option value="0"> 请 选择 </option> 
@ <?php foreach($papers as $paper) : ?> 
<option value="<?php echo $paper->id ?>"><?php echo $paper->paper_name ?></option> 
<?php endforeach ?> 
</select> 
© <span class="input-group-btn" onclick="startExam(<?php echo $paper->id ?>)"> 
<button type="submit" class="btn" style="background-color:#73CA33;"> 开 始 考试 </button> 
</span> 
</div> 


<script> 
// 跳 转 至 开始 考试 
function startExam(){ 
/获取 选中 的 试卷 ID 
var paper_id = $('‘#papers').val(); 
9 if(paper_id == 0X{ 
layer.msg(' 请 选择 试卷 ); 
return false; 
} 
$.ajax({ 
© url:'/check_paper’, 
type:'POST,', 
dataType:'json', 
data:{paper_id':paper_id}, 
success:function(res){ 
if(res.status == 1){ 
layer.msg(res.msg); 
© window.location.href = /exam/paper'+paper_id; 
Jelse{ 
layer.msg(res.msg); 
} 
} 
六 
return false; 


} 


</script> 


Ah 代码 贴 十 
@value=0: 请 选择 信息 。 
@ foreach: foreach 遍历 数据 ， 注 意 foreach 和 endforeach 是 一 对 标签 。 
@ onclick="startExam0": 单 击 事件 。 
@ paper id 一 0: 判断 是 否 选 择 试卷 。 
@ url/check paper: 异步 提交 到 check paper 路 由 。 
@ windowlocation href: 页 面 跳 转 。 


上 述 代码 中 ， 单 击 “ 开 始 考试 ”按钮 后 ， 使 用 Ajax 异步 提交 到 “/check _ paper” 路 由 ， 该 路 由 对 应 
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Home 控制 器 的 checkPaper0 方 法 ， 具 体 代 码 如 下 : 


倒 程 05 代码 位 置 资源 包 \TM\04\Exam\app\controllers\HomeController.php 


// 检 测试 卷 
public function checkPaper(){ 
$paper_id =$_POST[paper_id]; 
$questions = Questions::where('‘paper_id',$paper_id)->get()->toArray(); 
if(I$questions)\{ 
$res['status] = 0; 
$res['msg'] = ' 该 试卷 下 暂 无 考题 ， 请 选择 其 他 试卷 ; 
Jelse{ 
$res['status] = 1; 
$res[msg'] = ' 正 在 加 载 试卷 ， 准 备考 试 '; 


} 
echo json_encode($res); // 将 数组 转化 为 json 格式 
} 





当 未 选择 考题 单 击 “ 开 始 考试 ”按钮 时 ， 运 行 结果 如 图 4.16 所 示 。 当 试卷 下 没有 考题 ， 单 击 “ 开 
始 考试 ”按钮 时 ， 运 行 结果 如 图 4.17 所 示 。 试 卷 下 有 考题 运行 结果 如 图 4.18 所 示 。 





Wane mos a] 


考场 规定 








图 4.16 未 选择 考题 运行 结果 





成 二 可 光 。 作证 者 玖 会 品 


考场 规定 考场 提示 


让 汪 无 二 ， 汉 壬 二 他 所 。 蜂 














图 4.17 试卷 下 无 考题 运行 结果 
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图 4.18 试卷 下 有 考题 运行 结果 
4.7 在 线 考试 模块 设计 


在 线 考试 模块 概述 


在 线 考试 模块 的 主要 功能 是 允许 考生 在 网 站 上 针对 指定 的 课程 进行 考试 。 在 该 模块 中 ， 考 生 首先 
需要 登录 到 本 系统 中 ， 选 择 考试 科目 ， 选 择 科目 下 的 试卷 ， 然 后 进入 考试 页 面 进行 答题 ， 当 考生 提交 
试卷 或 者 到 达 考试 结束 时 间 时 ， 系 统 将 自动 对 考生 提交 的 试卷 进行 评分 ， 并 给 出 最 终 考 试 成 绩 。 


4.7.2 在线 考试 模块 技术 分 析 


在 线 考试 模块 中 最 核心 的 功能 是 考试 计时 、 输 出 考题 、 对 答案 进行 判断 和 输出 考试 成 绩 。 考 试 计 
时 模块 通过 使 用 JavaScript 的 setIntervalO 函 数 每 秒 钟 执行 一 次 ， 判 断 是 否 到 达 考 试 时 间 。 在 输出 考题 
时 ， 从 questions 表 中 获取 所 有 考题 数据 ， 包 括 标题 、 得 分 和 选项 等 内 容 。 输 出 考试 成 绩 是 将 用 户 选 择 
的 答案 与 标准 答案 进行 对 比 ， 然 后 计算 所 有 正确 答案 的 总 分 数 。 


4.7.3 考试 计时 的 实现 过 程 


国 考试 计时 使 用 的 数据 表 : papers、questions 

1. 计算 剩余 时 间 

每 套 试卷 都 可 以 在 后 台 设 置 考试 时 间 ， 也 就 是 papers 表 中 total time 字段 的 值 (单位 是 秒 ) 。 每 隔 
1 秒 钟 ， 总 时 长 减 1， 并 将 剩余 时 长 填充 到 页 面 。 自 定义 一 个 jquery.timeout;js 文件 实现 该 功能 ， 具 体 
代码 如 下 : 


倒 程 06 ”代码 位 置 : 资源 包 \TM\04\Exam\app\public\js\jquery.timeout.js 


/倒计时 功能 
(function($j{ 
$.fn.timeout = function(options){ 
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options = $.extend({ 


@ "maxTime":60, /时 限 〈 秒 数 ) 
"onTimeOver":function()} /到 时 间 时 执行 的 函数 

}, options); 
var $thisObj = this; 

© var maxTime = options.maxTime; /1/ 倒 计时 〔 秒 数 ) 

© var timer = setInterval(update,1000); 1/1 秒 钟 更 新 一 次 

@ update(); /| 自动 更 新 第 一 次 
/自动 更 新 
function update(}{ 


if(maxTime>=0\{ 
var minutes = Math.floor(maxTime/60); /计算 分 钟 数 
var seconds = Math.floor(maxTime%60); 。 // 计 算 秒 数 

© S$thisObj.text(fill(minutes)+" 分 "+fill(seconds)+" 秒 "); 

--maxTime; 

Jelse{ 
clearlnterval(timer); 
options.onTimeOver(); // 调 用 函数 


} 


} 
// 不 足 10 位 自动 补 0 
function fill(s) { 
retumn s <10?'0'+s:s; 
> 
上 
DIQuery); 





代码 贴 二 
@ maxTime: 设置 默认 时 长 。 
@ options.maxTime: 考试 时 长 。 
@ setInterval: 按照 指定 的 周期 (以 毫秒 计 ) 来 调用 函数 或 计算 表达 式 。 
@update0: 自 定义 更 新 函数 ， 每 秒 钟 时 长 减 1， 并 将 总 秒 数 转化 为 “时 : 分 ”形式 。 
@ fill0: 自 定义 方法 ， 将 剩余 时 间 写 入 页 面 。 


运行 效果 如 图 4.19 所 示 。 





内 小 学 5 年 级 期 中 考试 数学 题 











图 4.19 考试 倒计时 效果 
2. 超时 自动 提交 试卷 
当 用 户 答题 时 间 超 过 考试 时 间 ， 系 统 自动 提交 试卷 。jquery.timeout.js 中 定义 了 一 个 onTimeOver 
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方法 ， 这 是 一 个 空 方法 ， 需 要 在 exam.php 文件 中 覆盖 该 方法 ， 具 体 代码 如 下 : 


倒 程 07 代码 位 置 ， 资源 包 \TM\04\Exam\app\views\home\exam.php 
$(".timeout").timeout({ 
// 考 试 时 间 (页面 刷 新 时 ， 时 间 会 重 置 。) 
@ "maxTime": maxTime, 
// 到 达 时 间 自 动 交卷 《如 果 浏 览 器 禁用 JavaScript， 此 功能 不 会 生效 ) 
"onTimeOver": function() { 
@ timeOver = true; 
alert(" 考 试 时 间 结 束 ， 系 统 自动 交卷 。"); 
目 $("#form-submit").submit(); /交卷 


站 
D); 


< 代 码 贴 二 
@ maxTime: 考试 时 长 。 
@ timeOver: 设置 标识 为 nue， 即 标识 已 经 超时 。 
@ submitO: 提交 表单 。 


运行 结果 如 图 4.20 所 示 。 
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图 4.20 自动 提交 试卷 效果 图 
3. 刷新 或 关闭 页 面 提示 


在 考试 期 间 用 户 刷新 或 关闭 页 面 会 视 为 放弃 考试 ， 所 以 在 用 户 刷新 或 关闭 页 面 时 ， 可 以 使 用 
beforeunload0 方 法 弹出 提示 信息 。 如 果 用 户 依然 选择 “离开 ”， 则 本 次 考试 结束 。 如 果 选 择 “ 留 下 ”， 
则 继续 考试 。 代 码 如 下 : 


倒 程 08 ”代码 位 置 ， 资源 包 \TM\04\Exam\app\views\home\exam.php 





/关闭 页 面前 提示 
$(window).on("beforeunload", function() { 

return "您 尚未 交卷 ! 此 操作 将 导致 您 的 回答 丢失 。"; 
D; 


4 
Pe 
里 ， 但 有 些 其 他 浏览 器 只 展示 它们 自 定义 的 信息 (如 谷歌 浏览 器 只 展示 它 自 定义 的 信息 : 系统 可 能 
不 会 保存 您 所 做 的 更 改 ) 。 没 有 赋值 时 ， 该 事件 不 做 任何 响应 。 


运行 效果 如 图 4.21 所 示 。 


他- 
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图 421 刷新 或 关闭 提示 
4.7.4 考题 页 面 的 实现 过 程 


1， 获取 考题 数据 


Exam 表 存 储 着 所 有 考题 信息 ， 为 获取 某 个 试卷 的 信息 以 及 考题 信息 ， 需 要 根据 试卷 ID 筛选 出 该 
试卷 的 考题 。 具 体 代 码 如 下 : 


倒 程 09 代码 位 置 ， 资源 包 \TM\04\Exam\app\controllers\HomeController.php 
pn 

* 考试 页 面 

* @param $paper_id: 试卷 ID 

学 





public function exam($paper_id) 
{ 
©@ $paper= Papers::find($paper_id); 
$questions = Questions::where(paper_id',$paper_id)->orderBy(order ,asc)->get(); 
@ Sthis->view = View:make(home.exam')->with(paper,$papen) 
->with('questions', $questions); 


} 





< 八代 码 贴 二 
@ find0: 根据 试卷 TD， 获取 试卷 信息 。 
@ with: 向 模板 中 赋值 多 个 变量 。 


2. 展示 考题 


在 考试 页 面 ， 由 于 单 选 和 多 选 题 的 展现 形式 不 同 ， 在 遍历 考题 数据 的 时 需要 判断 考题 类 型 。 如 果 
questions 表 中 type 字段 值 为 1 是 单 选 题 ， 单 选 题 类 型 为 “radio”; type 字段 值 为 2 是 多 选 题 ， 多 选 题 
类 型 为 “checkbox”。 此 外 ， 由 于 选项 信息 存储 在 questions 表 的 option 字段 中 ， 遍 历 每 道 题 时 ， 需 要 
拆 分 4 个 选项 。 关 键 代码 如 下 : 


倒 程 10 ”代码 位 置 : 资源 包 \TM\04\Exam\app\views\home\exam.php 
<?php foreach($questions as $question) : 
$i=0; 
© $options = explode(',",$question->options); 
© if($question->type == 1 X{ 
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$type = "radio' 
$char = ' 单 选 题 "; 
}else{ 
$type = 'checkbox'; 
$char = 多 选 题 '; 
$ 
?> 
<div class="question"> 
<tr class="success" > 
<th> 
<a name="a<?php echo $question->order ?>" style="padding-top: 61px;"></a> 
<h4 style="color: #000000;"> 
<span class="glyphicon glyphicon-paperclip text-danger"></span> 
<?php echo $question->order ?>. 
<?php echo $char ?> 
<?php echo $question->title ?> 
[<a href="javascript:void(0)" onclick="setFlag(<?php echo $question->order ?>)"> 
答题 标记 </a>] 
(本 题 <?php echo $question->score ?> 分 ) 
</h4> 
</th> 
</tr> 
<tr> 
<td style="padding-left: 40px"> 
<ol type='A' class="text-primary" style="font-size: 14px;"> 


© <?php foreach(range('A', 'D') as $char) : ?> 
<li> 
<div class="<?php echo $type ?>"> 
日 <label><input type="<?php echo $type ?>" 


value="<?php echo $char ?>" 


9 name="answers[<?php echo $question->id ?>]0"> 
9 <?php echo $options[S$i++] ?> 
</label> 
</div> 
< 中 > 
<?php endforeach ?> 
</ol> 
<br/> 
</td> 
</tr> 
</div> 
<?php endforeach?> 
< 代码 贴 十 


@explode0: 字符 串 拼 接 成 数组 。 

@ $question->type: 判断 题目 类 型 。 

@@ range('A','D'): 创建 一 个 包含 从 "A" 到"D" 之 间 的 元 素 范围 的 数组 。 
@ type: 输入 类 型 ， 值 为 “radio” 是 单 选 ， 值 为 “checkbox” 是 多 选 。 
@ name: 值 为 二 维 数组 . 

@ $options: 依次 获取 选项 信息 。 
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运行 结果 如 图 4.22 所 示 。 
号 2. 绊 选 是 ] x+2=8, 求 x=? [答题 标记 ]( 本 题 2 分 ) 
| 考试 各 AO1 
BO2 
全 iE 24 分 3 和 本 
号 3. [多 选 机 ] 以 下 哪 一 个 是 好 声音 导师 [ 答 要 标 训 ] ( 本 是 2 分 ) 
A 回避 关 
B 国庆 以 
C 目 周 # 奸 
D. 回 局 本 伦 
图 4.22 考试 考题 运行 效果 
3. 标记 答题 卡 


在 题目 较 多 情况 下 ， 用 户 很 可 能 出 现 漏 答 现象 ， 为 此 ， 我 们 设计 了 答题 卡 功能 。 当 用 户 每 答 完 一 
题 ， 可 以 单 击 题目 右 侧 的 “答题 标记 ”， 此 时 答题 卡 中 显示 该 题 的 背景 颜色 为 黄色 ， 即 提示 用 户 该 题 
已 经 答 完 。 当 再 次 单 击 “ 答 题 标 记 ”， 此 时 答题 卡 中 显示 该 题 的 背景 颜色 消失 。 实 现 该 功能 的 关键 代 
码 如 下 : 


倒 程 11 代码 位 置 , 资源 包 \TM\04\Exam\app\views\home\exam.php 


<tr class="success" > 
<th> 
<a name="a<?php echo $question->order ?>" style="padding-top: 61px;"></a> 
<h4 style="color: #000000;"> 
<span class="glyphicon glyphicon-paperclip text-danger"></span> 
<?php echo $question->order; ?>. 
[<?php echo $char; ?>] 
<?php echo $question->title ?> 
[<a href="javascript:void(0)" onclick="setFlag(<?php echo $question->order ?>)"> 答 题 标记 </a>] 
(本 题 <?php echo $question->score ?> 分 ) 
</h4> 
</th> 
</tr> 
/添加 删除 标记 
function setFlag(flag){ 
if ($("#dtk"+flag).hasClass('btn-info")}{ 
$("#dtk"+flag).removeClass('btn-info').addClass('btn-warning'); 
}else{ 
$("#dtk"+flag).removeClass('btn-warning').addClass('btn-info'"); 





} 
了 


运行 结果 如 图 4.23 所 示 。 
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图 4.23 设置 答题 标记 
4. 提交 试卷 


用 户 在 考试 时 间 内 答 完 题目 ， 可 以 单 击 “ 交 卷 ”按钮 提交 试卷 。 提 交 试 卷 前 ， 会 弹出 确认 提示 框 ， 
当 用户 单 击 “确定 ”按钮 ， 开 始 提交 试卷 ， 否 则 不 提交 试卷 。 实 现 该 功能 的 关键 代码 如 下 : 


倒 程 12 ”代码 位 置 ; 资源 包 \TM\04\Exam\app\views\home\exam.php 


<div class="panel-footer" > 
<p class="text-center"> 
<button type="button" class="btn btn-success btn-sm" onclick="submitGo();,"> 
<span class="glyphicon glyphicon-pencil" ></span> 交 卷 
</button> 


function submitGo()f 
layer.confirm(' 是 否 确定 提交 答卷 ,function(){ 
var end_time = new Date().getTime(); /获取 当前 时 间作 为 交卷 时 间 
S$(#end_time'’).val(end_time/1000); /毫秒 转化 为 秒 
S$("#form-submit").submit(); /提交 表单 
入 
和 





运行 效果 如 图 4.24 所 示 。 





4.24 表单 提交 的 提示 信息 
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4.7.5 ”分 数 统计 和 成 绩 保存 的 实现 过 程 


国 ”分 数 统计 和 成 绩 保存 使 用 的 数据 表 : questions、results 
在 线 考 试 模块 中 不 但 可 以 完成 考试 的 答题 和 判 卷 的 操作 ， 而 且 可 以 直接 对 考试 的 分 数 进行 统计 ， 
获取 考生 的 最 终 得 分 ， 将 考生 的 成 绩 保存 到 数据 库 中 。 考 试 分 数 统计 和 存储 考试 成 绩 的 关键 代码 
如 下 : 
倒 程 13 ”代码 位 置 ， 资源 包 \TM\04\Exam\app\controllers\HomeController.php 
public function result(X{ 





/YY 计算 分 数 **/ 
Stotal =0; 
$answers = isset($_POST['answers]) ? $_POST[answers'] : array(); /获取 答案 数组 
if(lempty($answers)}{ 
foreach($answers as $k => $answer){ 
$question = Questions::find($k); // 获 取 正 确 答案 
if($question->answers == implode(',', $answer)}{ // 判 断 答案 是 否 正 确 
S$total += $question->score; /统计 分 数 
} 
t 
} 
$paper = Papers::find($_POST[paper_id]); /查找 试卷 信息 
/* 写 入 result 表 **/ 
$result = new Results; 
Sresult>user id = $_SESSION[id]; /用 户 ID 
Sresult->username = $_SESSION['username'; /用 户 名 
Sresult->paper_id = $_POST[paper_id]; /试卷 ID 
Sresult->paper_name = $paper->paper_name; /试卷 名 称 
$result->score C= $total; /总 分 数 
Sresult->start_time = date(Y-m-d H:i:s,$_POST[start_time]); // 开 始 时 间 
ifl$_POST[end_time]j /| 系统 自动 提交 
$differ = $paper->total_time; /答题 用 时 
$result->end_time = date(Y-m-d H:is,$_POST[start_time] + $differ); /结束 时 间 
jelsef /用 户主 动 提交 
$differ =$_POST[end time] - $_POSTT'start_time':; /答题 用 时 
Sresult->end_time = date(Y-m-d H:is',$_POST[end_time]); /| 结束 时 间 
3 
$minites = intval($differ/60); /| 转化 为 分 
S$seconds = $differ % 60; /| 转化 为 秒 
S$use_time = $minites.' 分 .$seconds.' 秒 '; /拼接 用 户 考试 用 时 
Sresult->save(); /保存 成 绩 
/* 模 板 赋值 */ 


Sthis->view = View::make(home.result)->with(paper_name',$paper->paper_name) 
->with('start_time', $result->start_time) 
->with('use_time',$use_time) 
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->with('total", $total); 
} 


上 述 代 码 中 ， 用 户 提 交 的 每 一 道 题 的 答案 是 一 个 数组 ， 使 用 implode0 函 数 先 将 数组 拆 分 为 逗号 隔 
开 的 字符 串 , 然后 将 该 字符 串 和 question 表 中 的 答案 进行 对 比 。 如 果 相 同 , 则 令 $score 加 上 该 题 的 分 数 ， 
最 后 得 到 总 分 数 。 此 外 ， 需 要 注意 系统 自动 提交 答卷 和 用 户主 动 提交 答卷 时 ， 获 取 结 束 时 间 的 区 别 。 
运行 效果 如 图 4.25 所 示 。 





尊敬 的 "mr" : 您 于 2017-12-19 15:55:51 参与 ee 可] 考试 ， 
用 时 22 分 55 秒 ， 的 


4.25 ”统计 分 数 


4.7.6 查看 历史 成 绩 的 实现 过 


1. 获取 历史 成 绩 数据 


从 results 表 中 筛选 用 户 ID 为 当前 用 户 DD 的 所 有 数据 ， 然 后 根据 考试 开始 时 间 降 序 排序 ， 获 取 用 
户 历 史 成 绩 的 关键 代码 如 下 : 


倒 程 14 ”代码 位 置 ; 资源 包 \TM\04\Exam\app\controllers\HomeController.php 





/历史 成 绩 

public function history(){ 
S$user_id =$_SESSION['id]; /获取 用 户 ID 
Sresults = Results::where('user_id', $user_id)->orderBy('start_time','desc’)->get(); /查找 results 表 数 据 
Sthis->view = View::make('home.history')->with('results', $results); // 泻 染 模板 

} 





2. 显示 历史 成 绩 
在 历史 成 绩 页 面 ， 我 们 以 表格 的 形式 展示 试卷 名 称 、 开 始 时 间 、 交 卷 时 间 和 考试 分 数 4 个 字段 ， 
关键 代码 如 下 : 


倒 程 15 ”代码 位 置 : 资源 包 \TM\04\Exam\app\views\home\history.php 


<?php require(BASE_PATH.'"/app/views/partials/head.php'); ?> 
<?php require(BASE_PATH./app/views/partials/home_nav.php'); ?> 


<div class="container" style="padding-top: 20px;"> 


<img alt=” src="images/chaxun03.png" class="img-responsive"> 
</div> 
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<div class="container" style="padding-top: 20px;"> 
<img alt=" src="images/chaxun01.png" class="img-responsive"> 
</div> 
<div class="container > 
<table class="table table-bordered text-center"> 
<tr> 
<th class="text-center" style="background-color #F1FAEA"> 试 卷 名 称 </th> 
<th class="text-center" style="background-color: #F1FAEA"> 开 始 时 间 </th> 
<th class="text-center" style="background-color: #F1FAEA"> 交 卷 时 间 </th> 
<th class="text-center" style="background-color: #F1FAEA"> 考 试 分 数 </th> 





</tr> 
<?php foreach($results as $result) : ?> 
<tr> 
<td><strong><?php echo $result->paper_name ?></strong></td> 
<td><strong><?php echo $result->start_time ?></strong></td> 
<td><strong><?php echo $result->end_time ?></strong></td> 
<td><strong><?php echo $result->score ?></strong></td> 
</tr> 
<?php endforeach ?> 
</table> 
</div> 
运行 结果 如 图 4.26 所 示 。 


考生 成 绩 





[3 
斌 名 际 开始 本 间 EN 考 汪 分数 


小 中 5 年 加 期中 专 斌 现 宁 员 2017-12-19 15:55:51 日 





小 学 5 年 报 期 中 二 试 宁 颁 2017-12: 








4.26 ”考试 成 绩 单 运行 效果 


4.8 后 台 管 理 模块 设计 





4.8.1 后 台 管 理 模块 概述 


后 台 管 理 模块 主要 包括 考试 科目 管理 、 试 卷 管理 和 考题 管理 3 个 模块 ， 而 每 个 模块 都 包含 添加 、 
修改 和 删除 等 功能 。 后 台 管 理 模块 的 框架 如 图 4.27 所 示 。 
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新 赠 科 下 
修改 笠 忆 
其 附 科 本 


新 庙 考 是 
修改 者 时 
删除 者 村 


考试 科目 管理 | 一 一 试卷 管理 [一 一 考题 管理 





新 增 试卷 
修改 试卷 
删除 试卷 








图 4.27 后 台 管理 模块 框架 
后 台 管 理 模块 技术 分 析 


后 台 管理 模块 重点 在 于 厘清 考试 科目 管理 、 试 卷 管理 和 考题 管理 3 个 模块 之 间 的 关系 ， 以 及 如 何 
传递 DD。 例 如 ， 考 试 科目 管理 页 面 要 将 考试 科目 ID 传递 到 试卷 管理 页 面 。 而 试卷 管理 页 面 需 要 将 试 
卷 D 传递 到 考题 管理 页 面 中 。 我 们 通过 在 链接 地 址 中 添加 ID 的 方式 来 实现 ID 的 传递 。 


4.8.3 科目 管理 的 实现 过 程 


回 “科目 管理 使 用 的 数据 表 : subjects 

在 科目 管理 模块 中 ， 重 点 讲解 科目 列表 页 的 实现 。 科 目 列表 页 包含 所 有 的 科目 信息 ， 并 且 具 有 分 
页 功能 。 有 具体 代码 如 下 : 

倒 程 16 代码 位 置 ， 资源 包 \TM\04\Exam\app\controllers\SubjectsController.php 

class SubjectsController extends BaseController 


4.8.2 





public function index($page=1) 
{ 


Stotalltems ”= Subjects::count(); /总 条 数 
$itemsPerPage = 8; /每 页 显示 条 数 
$currentPage = $page; // 当 前 页 码 
@ S$urlPattern = "/admin/subjects/index/page/(:num)’; Jurl 类 型 
@ S$paginator = new Paginator($totalltems, $itemsPerPage, $currentPage, $urlPattern); // 实 例 化 分 页 类 
目 Soffset = ($page-1)*$itemsPerPage; // 设 置 开 始 位 置 
9 $subjects = Subjects::orderBy('created_at','desc')->limit($itemsPerPage)->offset($offset)->get(); 
Sthis->view = View::make('subjects.index') 
->with('subjects', $subjects) 
->with(paginator,$paginator); 
} 
< 代码 贴 十 


@ urlPattem: 分 页 的 trl 形式 。 

@ new Paginator0: 实例 化 分 页 类 ， 该 分 页 类 已 经 通过 Composer 引入 。 
目 $offset: 设置 分 页 的 开始 位 置 。 

@ offset0: 调用 分 页 方法 。 
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获取 到 数据 后 泻 染 模板 ， 模 板 页 面 代码 如 下 : 


倒 程 17 代码 位 置 资源 包 \TM\04\Exam\app\views\subjects\index.php 


<div class="main-right col-md-10 col-md-offset-2"> 
<div class="col-md-12"> 
<div class="panel panel-default "> 
<div class="panel-heading"> 
<div class="row"> 
<div class="col-sm-4"> 
<div class="input-button"> 
<a href="/admin/subjects/add"> 
<button class="btn btn-primary add" type="button"> 
<i class="glyphicon glyphicon-plus"></i>&nbsp; 新 增 
</button> 
</a> 
</div> 
</div> 
</div> 
</div> 
<div class="panel-body"> 
<table class="table table-bordered tb-hover tb-bg" style="margin-bottom: 5px; "> 
<thead> 
<tr> 
<td>ID</td> 
<td> 科 目 名 称 </td> 
<td> 创 建 时 间 </td> 
<td> 更 新 时 间 </td> 
<td class="text-center"> 操 作 </td> 
</tr> 
</thead> 
<tbody> 
<?php foreach ($subjects as $subject) : ?> 
<tr> 
<td><?php echo $subject->id; ?></td> 
<td><?php echo $subject->subject_name; ?></td> 
<td><?php echo $subject->created_at; ?></td> 
<td><?php echo $subject->updated_at; ?></td> 
<td style="width:250px"> 
人 <a href="/admin/papers/index/subject/<?php echo $subject->id ?>"> 
<button class="btn btn-warning " type="button"> 试 卷 管理 </button> 
</a> 
<a href="/admin/subjects/edit/<?php echo $subject->id ?>"> 
<button class="btn btn-info " type="button"> 编 辑 </button> 
</a> 
<a href="/admin/subjects/delete" class="del" 
value="<?php echo $subject->id ?>"> 
<button class="btn btn-danger " type="button"> 删 除 </button> 
</a> 
</td> 
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</tr> 
<?php endforeach; ?> 
</tbody> 
</table> 
</div> 
<|-- 分 页 --> 
<div class="paginator" style="padding-right: 10px"> 
© <?php echo $paginator;?> 
</div> 
</div> 
</div> 
</div> 
二 代码 贴 十 


@ herf; 跳 转 到 试卷 管理 页 面 。 
@ $paginator: 分 页 样式 。 


考试 科目 管理 的 运行 效果 如 图 4.28 所 示 。 




















考试 系统 遇 El 
”| eT] 
se Ec 











图 428 考试 科目 管理 页 运行 效果 
4.8.4 试卷 管理 的 实现 过 程 


在 考试 科目 管理 页 面 ， 单 击 相应 科目 的 “考题 管理 ”按钮 ， 进 入 到 试卷 管理 页 面 。 试 卷 管理 页 面 
功能 与 科目 管理 页 面相 似 ， 运 行 效果 如 图 4.29 所 示 。 单 击 试卷 页 面 上 方 的 “新 增 ” 按 钮 ， 进 入 新 增 试 
卷 页 面 ， 由 于 当前 科目 是 “数学 ”， 所 以 在 新 增 试卷 页 面 ， 所 属 科目 默认 为 “数学 ”， 运 行 效果 如 
图 4.30 所 示 。 在 考试 科目 管理 单 击 “ 编 辑 ” 按 钮 ， 运 行 效果 如 图 4.31 所 示 。 单 击 “ 删 除 ” 按 钮 ， 提 示 
“确认 删除 该 试卷 吗 ? ”， 当 单 击 “确定 ”按钮 时 ， 首 先 判断 该 试卷 下 是 否 有 考题 ， 如 果 有 考题 ， 提 
示 “ 该 试卷 下 存在 考题 ， 请 先 删除 该 试卷 的 考题 ”， 运 行 效果 如 图 4.32 和 图 4.33 所 示 。 在 试卷 管理 模 
块 ， 我 们 重点 讲解 试卷 的 编辑 和 删除 功能 。 
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iD。 试 生 名 疗 所 感 科目 。 出 加 才 策 。 到 时 时间 更 条 时间 
14 。 4 学? 年 顷 基 中 专 局 各 字 量 a 过 各 2017-12.08 143331 2017-1208 143331 
13 。 4 学 1 年 天 期 中 考 放 数字 是 到 学 老 马 2017-1208143318 2017-12.08143318 
12 小 学 6 年 级 期 中 奥数 大 到 老 刘 2017-1208 141905 2017-12.0814.3247 
11 小 学 5 年 委 天 中 奥 笋 训 守 a 老 关 2017-12.08141857 2017-12.08143234 
10 小 学 4 征地 中 奥 至 珊 委 mF 二 李 20171208134832 2017-12.08143213 
9 。 4 宁 5 年 及 基 中 负数 育 究 地 老 时 2017-1208134637 。。 2017-1208143141 
小 学 6 年 生 蓝 中考 记 由 字 是 效 学 老 王 2017-12.08 134609 2017-12.08 143104 
6 小子 5 年 级 期中 考试 政子 开 wr | 20171208 134600 20171208143031 





图 4.29 试卷 管理 页 面 运行 效果 




















二 本 达 将 信 息 人 下 放 人 
所 属 科 日 所 民有 日 
a Ea 
这 寡 名 称 [1 
学 2 生肖 期 中 考试 了 学 是 
ED] 关 雪 时 间 
意 位 是 秒 , 如 1 二 写 350 1900 
出 题 人 出 明和 人 
Ee 
四 加 
4.30 新 增 试卷 页 面 效果 4.31 编辑 试卷 页 面 效果 
Ea x 
全 隐 这 二 7? 
PE 
4.32 ”删除 试卷 的 提示 图 4.33 ”提示 先 删除 试题 


1. 编辑 试卷 


如 果 要 编辑 试卷 ， 首 先 需 要 根据 试卷 的 ID 进入 相应 试卷 的 编辑 页 面 ， 进 入 编辑 页 面 后 ， 获 取 页 面 
内 容 ， 然 后 修改 内 容 ， 最 后 提交 表单 。 编 辑 页 面 获取 内 容 的 代码 如 下 : 


倒 程 18 ”代码 位 置 : 资源 包 \TM\04\Exam\app\views\papers\edit.php 


<div class="main-right col-md-10 col-md-offset-2"> 
<div class="col-md-12"> 
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<div class="panel panel-default "> 
<div class="panel-heading"> 


修改 试卷 信息 
</div> 
<div class="panel-body"> 





<form id="submit-form" action="/admin/papers/update" method="post" class="col-md-6" > 
<input type="hidden" id="subject_id" name="subject_id" 
value="<?php echo $subject->id ?>"> 
<div class="form-group"> 
<label for="subject_name"> 所 属 科目 </label> 
<input type="text" class="form-control" id="subject_name" readonly 
name="subject_name" value="<?php echo $paper->subject_name ?>"> 
</div> 
<div class="form-group"> 
<label for="paper_name"> 试 卷 名 称 </label> 
<input type="text" class="form-control" id="paper_name" 
name="paper_name" value="<?php echo $paper->paper_name ?>"> 
</div> 
<div class="form-group"> 
<label for="paper_name"> 答 题 时 间 </label> 
<input type="text" class="form-control" id="total_time" 
name="paper_name" value="<?php echo $paper->total_time ?>" 
placeholder=" 单 位 是 秒 ， 如 1 小 时 填写 3600"> 
</div> 
<div class="form-group"> 
<label for="teacher_name"> 出 题 人 </label> 
<input type="text" class="form-control" id="teacher_name" 
name="teacher_name" value="<?php echo $paper->teacher_name ?>"> 


</div> 
<button id="submit” class="btn btn-primary"> 提 交 </button> 
</form> 
</div> 
</div> 
</div> 
</div> 
<script> 


S$(#submit').click(function(){ 

var paper_id = <?php echo $paper->id ?>; 

/验证 

var paper_name = $("#paper_name").val(); 

if(IcheckField(#paper_name',' 请 输入 试卷 名 称 !)){ 
return false; 

} 

var teacher_name = $("#eacher_name").val(); 

if(IcheckField( 术 eacher_name',' 请 输入 试卷 名 称 !)){ 
return false; 


} 
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/lAjax 提交 
var url = $(#submit-form ').attr(action'); 
varjump = "/admin/papers/index/subject/"+<?php echo $paper->subject_ id ?>; 


$.ajax({ 
url:url, 
type:'POST', 
dataType:json', 
data:{'id':paper_id,'paper_name':paper_name,'teacher_name':teacher_name}, 
Success:function(res)f 
if(res.status == 1X{ 
layer.msg(res.msg); 
window.location.href = jump; 
jelsef 
layer.msg(res.msg); 
} 
} 
六 
return false; 
六: 
/检测 字段 是 否 为 空 


function checkField(id,message){ 
if($(id).val() == "X 
layertips(message, id, {time: 2000, tips: 2}); 


S$(id).focus(); 
return false; 
} 
return true; 
} 
</script> 





当 用 户 单 击 “ 提 交 ” 按 钮 ,会 使 用 Ajax 异步 提交 到 PapersController 类 的 update0 方 法 , 在 update() 
方法 中 更 新 papers 表 数 据 。update() 方 法 的 具体 代码 如 下 : 


倒 竹 19 ”代码 位 置 ， 资源 包 \TM\04\Exam\app\controllers\PapersController.php 
public function update() 


{ 
$id=$_POST[id]; // 获 取 ID 
$subjects = Papers::find($id); /实例 化 Paper 类 
$subjects->paper_name = $_POST[paper_name'];  // 获 取 试 卷 名 称 
S$subjects->total time = $_POSTTtotal time'; // 获 取 试 卷 名 称 
$subjects->teacher_name = $_POST[teacher_name']; /获取 老师 名 称 
if($subjects->save()){ /保存 数据 


$res['status'] = 1; 

Sres['msg'] = ' 操 作成 功 '; 
}else{ 

Sres['status'] = 0; 
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S$res[msg'] = ' 操 作 和 失败 
} 
echo json_encode(S$res); /将 数组 转化 为 json 格式 
} 


2. 删除 试卷 


首先 看 一 下 试卷 列表 页 面 中 的 “删除 ”按钮 。 当 单 击 “删除 ”按钮 时 ， 触 发 单 击 事件 ， 关 键 代码 
如 下 : 


倒 程 20 ”代码 位 置 ， 资源 包 \TM\04\Exam\app\views\papers\index.php 
<script> 
S$(function() { 
$('.del').on('click',function() { 
varid = $(this).attr('value'); 
var url = $(".del").attr('href ); 
// 弹 出 确认 框 
layerconfirm(' 确 认 删 除 该 试卷 吗 ?,function(){ 
$.ajax({ 
urlurl， 
type:"POST , 
dataType:json'， 
data:fid"id}, 
Success:function(res){ 
if(res.status == 1X{ 
layer.msg(res.msg); 
window.location.reload(); 
Jelse{ 
layer.msg(res.msg); 
四 
} 
入 
»; 
return false; 
了) 
)) 
</script> 


上 述 代码 中 ， 使 用 Ajax 异步 提交 到 PaperController 类 下 的 delete0 方 法 ， 具 体 代码 如 下 : 


倒 程 21 代码 位 置 : 资源 包 \TM\04\Exam\app\controllers\PapersController.php 


public function delete() 
‘ 


$id = $_POSTTidT: // 获 取 ID 
$paper = Papers::find($id); JISubject 调用 find() 方 法 
/* 判 断 试卷 是 否 有 考题 */ 


S$questions = Questions::where('‘paper_id',$id)->get(); 
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@ if($questions->toArray()X{ 
$res['status] = -1; 
S$res['msg'] = ' 该 试卷 下 存在 考题 ， 请 先 删 除 该 试卷 的 考题 '; 


@ echo json_encode(S$res); /将 数组 转化 为 json 格式 
die; 
} 
if($paper->delete()}X{ 1/ 删 除数 据 


$res[status] = 1; 
S$res['msg1] = ' 操 作成 功 '; 
}else{ 
$res['status] = 0; 
$res[msg'] = ' 操 作 失 败 '; 
} 
echo json_encode($res); /将 数组 转化 为 json 格式 


} 


< 代码 贴 十 
@ toArray0: 转化 为 数组 形式 。 
@ json_encode0: 将 数组 转化 为 json 格式 。 


4.8.5 考题 管理 的 实现 过 程 


在 试卷 管理 页 面 ， 单 击 相应 试卷 的 “考题 管理 ”按钮 ， 进 入 到 考题 管理 页 面 。 考 题 管理 页 面 功能 
与 科目 管理 页 面相 似 ， 运 行 效果 如 图 4.25 所 示 。 本 节 重 点 介绍 一 下 添加 考题 功能 。 

考题 共有 两 种 类 型 ， 单 选 题 和 多 选 题 。 对 于 单 选 题 ， 只 有 一 个 标准 答案 ， 对 于 多 选 题 ， 有 一 个 或 
多 个 答案 。 后 台 管 理 员 在 添加 考题 时 ， 需 要 选择 考题 的 类 型 ， 并 且 能 够 自动 切换 单 选 题 和 多 选 题 ， 然 
后 在 对 应 选项 后 选中 单 选 按钮 或 复 选 框 选 择 答案 。 我 们 使 用 JavaScript 来 实现 该 功能 。 关 键 代码 如 下 : 


倒 程 22 代码 位 置 ， 资源 包 \TM\04\Exam\app\views\questions\add.php 





$(document).ready(function() { 


// 切 换 单 选 或 多 选 
S$("#Htype").bind("change",function(}{ /切换 select 下 拉 菜单 选项 
if($(this).val()==1X{ // 如 果 值 为 1， 表示 单 选 
$(".single").show(); /显示 单 选 内 容 
S$(".multiple").hide(); // 隐 藏 多 选 内 容 
} 
else{ // 否 则 为 多 选 
S$(".single").hide(); /隐藏 单 选 内 容 
$(0.multiple").show(); /显示 多 选 内 容 


六 
D); 


选择 单 选 题 的 运行 效果 如 图 4.34 所 示 ， 选 择 多 选 题 的 运行 效果 如 图 4.35 所 示 。 
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境 号 斌 天 信息 声 写 坛 扫 圭 息 
所 局 试卷 小 学 5 年 航 关 中 考试 数学 本 i 小 学 5 年 级 月 中 考试 数 子 三 
是 号 : 8 分 数 : 2 要 3: 9 男 袁 : 
题目 : 个 翌 形 的 上 许 、 下 属 部 不 专 户 扩 大 到 原 幸 的 ? 信 它 的 而 二 (- 宣 目 : X 的 平方 =4 ， 求 X 的 什 了 
村 单 选 重 至 : 多 选 
篷 艳 :上 A 不 交 迄 项 : A 
日 | 扩大 2 售 晤 5 回 
可 we 6 
四 
图 4.34 单 选 题 运 行 效果 图 4.35 多 选 题 运行 效果 


填写 完 考题 后 ， 单 击 “ 提 交 ” 按 钮 ， 需 要 验证 提交 的 考题 信息 。 例 如 ， 用 户 是 否 填 写 了 题 号 、 题 
目 、 选 项 和 答案 等 内 容 。 此 外 ， 还 需要 获取 考题 答案 ， 例 如 单 选 题 的 答案 是 “B”， 多 选 题 的 答案 是 


“AB” 


， 然 后 将 所 有 信息 写 入 questions 表 。 关 键 代 码 如 下 : 


倒 程 23 代码 位 置 ， 资源 包 \TM\04\Exam\app\views\questions\add.php 





S$(#submit').click(function(}{ 


// 验 证 题 号 、 分 数 、 题 目 
if(!IcheckField('#order',' 请 输入 题 号 ")X( 
return false; 


} 
if(!checkField('#score',' 请 输入 分 数 ")X{ 
return false; 


} 
if(!icheckField('#title',' 请 输入 题目 ")){ 
return false; 
} 
Var obj; 
var answers ="; 
var options ="; 
if($(#type').val() == 1 
obj = $(.single input[name="option"]'); 


answers = $('input[type="radio"] [name="answer"]:checked').val(); 1/ 获取 答案 
}else{ 

obj = $('.multiple input[Iname="option"]'); 

/获取 答案 

$('input[type="checkbox"][name="answer"]:checked').each(function(X{ 

answers = answers + ,+ $(this).val(); 

入 

answers = trim(answers); // 去 除 第 一 个 逗号 
} 
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Var loop = true; 
obj.each(function(){ 
/检测 选项 内 容 是 否 为 空 
if(1$(this).valO)X 
layertips(" 请 填写 选项 内 容 ', $(this), {time: 2000, tips: 2}); 
loop = false; 
return false; 
}else{ 
// 获 取 选 项 
options = options + "+ $(this).val(); ”// 结 果 示例 : ， 选 项 A 内 容 ， 选 项 B 内 容 
console.log(options); 
入 
if(loop == falsej{ 
return false; 


} 
options = trim(options); // 去 除 第 一 个 喜 号 ， 结 果 示 例 : 选项 A 内 容 ， 选 项 B 内 容 
if(lanswersX{ 

layertips(' 请 在 右 侧 勾 选 答案 ', #items', {time: 2000, tips: 2}); 

return false; 


} 


/Ajax 提交 
var paper_id = <?php echo $paper->id ?>; 
var paper_name = $(‘#paper_name').val(); 
var order = $(#order).val(); 
Var Score = $('‘#score').val(); 
var title = $('#title').val(); 
vartype = $(‘#type').val(); 
var url = $(#submit-form ).attr(action ); 
var jump = "/admin/questions/index/paper/"+paper_id; 
© $3.ajax({ 
url:url, 
type:'POST,, 
dataType:'json', 
data:{paper_id':paper_id,'paper_name':paper_name,'order’:order,'score':score, title':title, 
'type':type,'options':options,'answers'-answers 
站 l/submit-form 是 表单 的 id 
Success:function(res){ 


if(res.status == 1){ 
layer.msg(res.msg); 
window.location.href = jump; 

Jelse{ 
layer.msg(res.msg); 

3 

3 
入 
return false; 


J; 
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/检测 字段 是 否 为 空 
function checkField(id,message){ 
if($(id).val() == "YX 
layertips(message, id, {time: 2000, tips: 2}); 
$(id).focus(); 
return false; 


} 


return true; 


} 

/去除 第 一 个 字符 

function trim(strX{ 
str = strsubstring(1,strlength); /去 除 第 一 个 字符 
return str; 


} 


< 代码 贴 二 
@ checkField0: 检查 字段 是 否 为 空 ， 如 果 为 空 ， 使 用 Layer 弹 层 插件 提示 错误 信息 。 
@ $(#type').val0: 获取 题 型 的 值 ， 如 果 为 1， 则 是 单 选 题 ; 如 果 为 2， 则 是 多 选 题 。 
目 answers: 获取 单 选 题 答案 。 
@ cach: 遍历 多 选 题 答案 。 
@ trim: 去 除 多 选 题 第 一 个 过 号 ， 将“.A.B” 更 改 为 “A.B”。 
@ $.ajax: 异步 提交 表单 数据 。 


提交 表单 时 ， 如 果 没有 选择 答案 ， 运 行 效果 如 图 4.36 所 示 。 


十 亏 试 时 全 局 

所 属 试 乞 : 。。 小 学 < 毕 柜 好 中 专 试 对 学 是 
是 号: 分 数 4 
缀 目 : X 的 平方 <4 ， 求 X 的 值 ? 
和 E22 





图 4.36 表单 提交 验证 


当 用 户 填写 完成 试题 内 容 ， 单 击 “ 提 交 ” 按 钮 ， 使 用 Ajax 异步 提交 到 QuestionsController 控制 器 
下 的 store0 方 法 。 具 体 代 码 如 下 : 


@ 
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倒 程 24 代码 位 置 资源 包 \TM\04\Exam\app\controllers\QuestionsController.php 


public function store() 


{ 














$questions = new Questions; /实例 化 Questions 类 
$questions->paper_id = $_POST[paper_id]; // 获 取 试 卷 ID 
$questions->paper_name = $_POST[paper_name]; /获取 试卷 名 称 
$questions->order = $_POST[order]; 1/ 获取 考题 序号 
$questions->score = $_POST[score]; 
$questions->title = $_POSTTtitle]; 1/ 获取 考 | 题 
$questions->type = 9$_POoST[type]; /获取 考题 类 型 
$questions->options = $_POST[options"; /获取 考题 选项 
S$questions->answers = $_POST[answers']; // 获 取 考 题 答案 
if($questions->save()X{ /保存 数据 

$res['status] = 1; 

$res[msg] = ' 操 作成 功 '; 
jelsef 

$res['status] = 0; 

$res[msg'] = ' 操 作 失 败 '; 
} 
echo json_encode($res); /将 数组 转化 为 json 格式 

} 





4.9 开发 技巧 与 难点 分 析 


4.9.1 Composer 常用 命令 


在 开发 过 程 中 ， 我 们 使 用 Composer 对 组 件 进行 管理 ， 由 于 Composer 运行 在 命令 行 中 ， 所 以 需要 
了 解 一 些 常用 Composer 命令 。 常 用 命令 如 下 。 
composer list; 显示 所 有 命令 。 
composer show: 显示 所 有 包 信息 。 
composer install: 在 composerjson 配置 中 添加 依赖 库 之 后 运行 此 命令 安装 。 
composer create-project laravel/laravel Laravel -prefer-dist“5.1.*”: 创建 项 目 。 
composer search packagename: 搜索 包 。 
composer update: 更 新 所 有 包 。 
composer update monolog/monolog: 更 新 指定 包 。 
composer remove monolog/monolog: 移 除 指定 的 包 。 
composer require monolog/monolog: 添加 指定 包 。 
composer require monolog/monolog:1.19: 添加 指定 包 和 版 本 。 
composer require monolog/monolog=1.19。 
composer require monolog/monolog 1.19。 
composer dump-autoload: 如 果 更 新 了 composerjson， 需 要 更 新 autoload。 
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4.9.2 ”中断 jQuery 的 each() 方 法 


在 后 台 添加 考题 模块 中 ， 提 交 考 题 前 ， 需 要 验证 考题 选项 是 否 为 空 。 我 们 使 用 jQuery 的 each0 方 法 
遍历 选项 ， 如 果 发 现 一 个 选项 为 空 ， 中 断 循环 并 弹出 提示 信息 。 可 以 在 each0 方 法 内 ， 使 用 return false 
来 中 断 循环 。 值 得 注意 的 是 ， 虽 然 循 环 中 断 了 ， 但 是 程序 会 继续 执行 each0 方 法 外 的 代码 。 为 了 解决 这 个 
问题 ， 可 以 设置 一 个 全 局 变量 loop， 初 始 值 为 tue。 当 中 断 循环 时 ， 设 置 loop 值 为 false。 具 体 代码 如 下 : 


Var loop = true; 
obj.each(function(){ 
/检测 选项 内 容 是 否 为 空 
if(!$(this).val()X{ 
layertips(' 请 填写 选项 内 容 ', $(this), ftime: 2000, tips: 2}); 
loop = false; 
return false; 
}else{ 
/获取 选项 
options = options + "+ $(this).val(); /结果 示例 : ， 选 项 A 内容， 选项 B 内 容 
} 


D); 
if(loop == false){ 
return false; 


} 





4.10 Ajax 无 刷新 技术 专题 


4.10.1 Ajax 概述 


Ajax 是 Asynchronous JavaScript and XML 的 缩写 ， 意 思 是 异步 的 JavaScript 与 XML 。 

Ajax 并 不 是 一 种 新 技术 ， 或 者 说 它 不 是 一 种 技术 ， 实 际 上 ， 它 是 将 JavaScript、XHTML 和 CSS、 
DOM、XML 和 XSTL、XMLHttpRequest 等 编程 技术 以 新 的 强大 方式 组 合 而 成 的 ， 可 以 让 开发 人 员 构 
建 基于 PHP 技术 的 Web 应 用 ， 并 打破 了 使 用 页 面 重 载 的 惯例 。Ajax 包含 以 下 方面 : 

XHTML 和 CSS 技术 实现 标准 页 面 。 

Document Object Model 技术 实现 动态 显示 和 交互 。 
XML 和 XSLT 技术 实现 数据 的 交换 和 维护 。 
XMLHttpRequest 技术 实现 异步 数据 接收 。 
JavaScript 绑 定 和 处 理 所 有 数据 。 

Ajax 是 一 种 运用 浏览 器 的 技术 ， 它 可 以 在 浏览 器 和 服务 器 之 间 得 到 异步 通信 机 制 进行 数据 通信 ， 
从 而 允许 浏览 器 向 服务 器 获取 少量 信息 而 不 是 刷新 整个 页 面 。 


4.10.2 Ajax 的 优点 
Ajax 是 使 用 客户 端 脚本 与 Web 服务 器 交换 数据 的 Web 应 用 开发 方法 。 这 样 ，Web 页 面 不 用 打 断 
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交互 流程 进行 重新 加 载 ， 就 可 以 动态 地 更 新 。Ajax 的 优点 介绍 如 下 
(1) 减轻 服务 器 的 负担 。 
Ajax 的 原则 是 “ 按 需 取 数 据 ?， 可 以 最 大 程度 地 减少 元 余 请 求 ， 从 而 减轻 对 服务 器 造成 的 负担 。 
(2) 无 刷新 更 新 页 面 ， 减 少 用户 心 理 和 实际 的 等 待 时间 。 
“ 按 需 取 数 据 ” 的 模式 减少 了 数据 的 实际 读 取 量 。 如 果 说 重 载 的 方式 是 从 一 个 终点 回 到 原点 再 到 另 
一 个 终点 的 话 〈 如 图 4.37 所 示 )， 那 么 Ajax 就 是 以 一 个 终点 为 基点 到 达 另 一 个 终点 (如 图 4.38 所 示 )。 
@ 
S 
和 <e 多 


8) © EN Ew 
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图 437 ” 重 载 方式 图 4.38 Ajax 方式 


其 次 ， 即 使 要 读 取 较 大 的 数据 ， 也 不 会 出 现 白 屏 的 情况 。Ajax 使 用 XMLHTTP 对 象 发 送 请 求 并 得 
到 服务 器 响应 ， 在 不 重新 载 入 整个 页 面 的 情况 下 用 JavaScript 操作 DOM 最 终 更 新 页 面 ， 所 以 在 读 取 数 
据 的 过 程 中 ， 用 户 所 面 对 的 不 是 白 屏 ， 而 是 原来 的 页 面 状态 ， 页 面 只 有 在 接收 到 全 部 数据 后 才 更 新 相 
应 部 分 的 内 容 ， 而 这 种 更 新 也 是 瞬间 的 ， 用 户 几 乎 感觉 不 到 。 

(3) 带 来 更 好 的 用 户 体验 。 

(4) 把 部 分 服务 器 负担 的 工作 转交 给 客户 端 ， 利 用 客户 端 闲置 的 能 力 来 处 理 任务 ， 从 而 减轻 服务 
器 和 带宽 的 负担 ， 节 约 空间 和 宽带 租用 成 本 。 

(5) 可 以 调用 外 部 数据 。 

(6) 是 一 种 基于 标准 化 并 被 广泛 支持 的 技术 ， 不 需要 下 载 插件 或 者 小 程序 。 

(7) 进一步 促进 Web 页 面 展现 形式 与 数据 的 分 离 。 


4.10.3 Ajax 的 工作 原理 


传统 的 Web 模式 强制 用 户 进 入 “提交 /等 待 /重新 显示 ”网 页 ， 用 户 的 动作 总 是 与 服务 器 进行 同步 思 
考 ， 客 户 在 网 页 上 的 操作 转化 为 HTTP 请 求 传 回 服务 器 ， 而 服务 器 接收 请 求 以 及 相关 数据 ， 然 后 解析 数 
据 并 将 其 发 送 给 相应 的 处 理 单元 后 ， 将 返回 的 数据 转 成 HTML 页 返还 给 客户 。 而 当 服 务 器 处 理 数 据 时 ， 
用 户 只 能 等 待 ， 每 一 步 操作 都 需要 等 待 服务 器 返回 新 的 网 页 。 由 于 每 次 应 用 的 交互 都 需要 向 服务 器 发 送 
请 求 ， 应 用 的 响应 时 间 就 依赖 于 服务 器 的 响应 时 间 ， 这 就 导致 了 用 户 页 面 的 响应 比 本 地 应 用 慢 得 多 。 

运用 了 Ajax 技术 的 Web 应 用 模型 , 它 的 工作 原理 相当 于 在 客户 端 和 服务 器 端 之 间 添 加 了 一 个 中 间 
层 ， 称 为 Ajax 引擎 (采用 JavaScript 编写 ， 通 常 在 一 个 隐藏 的 框架 中 )， 实 现 了 与 服务 器 进行 异步 思考 
的 通信 能 力 ， 从 而 使 用 户 从 请 求 /响应 的 循环 中 解脱 出 来 ， 向 服务 器 发 出 异步 请 求 ， 也 就 是 不 用 等 待 服 
务 器 的 通信 。 所 以 用 户 不 用 再 打开 一 个 空白 窗口 ， 等 待 服务 器 完成 后 再 进行 响应 。Ajax 应 用 可 以 仅 向 
服务 器 发 送 并 取 回 必需 的 数据 ， 它 使 用 SOAP 或 其 他 一 些 基 于 XML 的 Web Service 接口 ， 并 在 客户 端 
采用 JavaScript 处 理 来 自 服务 器 的 响应 。 因 为 在 服务 器 和 浏览 器 之 间 交 换 的 数据 大 量 减少 ， 所 以 Web 
站 点 看 起 来 是 即时 响应 的 。 同 时 很 多 的 处 理工 作 可 以 在 发 出 请 求 的 客户 端 机 器 上 完成 ， 所 以 Web 服务 


器 的 处 理 时 间 也 减少 了 。 
四 


引入 Ajax 的 Web 模型 与 传统 的 Web 模型 比较 如 图 4.39 所 示 。 
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《传统 Web 应 用 模型 > 





Ajax Web 应 用 模型 > 





图 4.39 传统 Web 应 用 模型 与 Ajax Web 应 用 模型 的 比较 
4.10.4 Ajax 的 工作 流程 


使 用 Ajax, 用 户 可 以 创建 接近 本 地 桌面 应 用 的 更 直接 、 更 可 用 、 更 丰富 、 更 动态 的 Web 用 户 界面 。 
Ajax 内 部 的 工作 流程 如 图 4.40 所 示 。 


一 


Web 服 务 器 格 请 求 使 用 HTML 和 JavaScript 响 应 





Intemet 用 户 Web 客户 端 Web 服 务 器 














图 4.40 Aijax 内 部 工作 流程 图 
区 
4.11 本 章 总 结 


至 此 ， 一 个 完整 的 在 线 考试 系统 已 经 全 部 完成 。 在 程序 的 开发 过 程 中 ， 使 用 组 件 化 方式 快速 实现 
特定 功能 。 同 时 ， 为 了 增加 交互 性 ， 系 统 中 应 用 了 Ajax 技术 。 和 希望 读者 能 认真 学 习 ， 并 做 到 融会 贯通 。 
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物流 配送 信息 网 


(Apachet+PHP+MySQL S.5 实现 ) 


物流 信息 化 是 指 物流 企业 运用 现代 信息 技术 对 物流 过 程 中 产生 的 
全 部 或 部 分 信息 进行 和 采集、 分 类 、 传 递 、 汇 总 、 坦 询 等 一 系列 处 理 活 
动 ， 以 实现 对 货物 流动 过 程 的 控制 ， 从 而 降低 成 本 、 提 高 效益 的 管理 
活动 。 

物流 企业 信息 化 的 目的 就 是 要 满足 企业 自身 管理 的 需要 和 不 同类 
型 企业 在 物流 业务 外 包 过 程 中 对 信息 交换 方 的 要 求 ， 也 就 是 通过 建设 
物流 信息 系统 ， 提 高 信息 流转 效率 ， 降 低 物 流 运作 成 本 。 

目前 我 国正 处 于 全 面 推进 信息 化 的 进程 之 中 ， 所 以 物流 领域 的 信 
息 化 已 成 为 一 个 必然 。 物 流 信息 化 将 成 为 现代 物流 的 灵 殊 ， 将 是 现代 
物流 发 展 的 必 有 经 之 路 。 

本 章 开发 的 物流 配送 信息 网 就 是 为 了 增强 企业 的 市 场 竞 争 力 ， 加 
快 企业 的 信息 化 进程 ， 使 企业 在 市 场 竞 争 中 立 于 不 败 之 地 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

MW 物流 配送 系统 开发 的 基本 过 程 

MW 如何 创建 MySQL 数据 库 的 存储 过 程 

”如 何 通过 MySQL 数据 库存 储 过 程 实现 用 户 登 录 

MW 如何 实 现 数据 库 中 数据 的 模糊 查询 技术 

由 如何 实现 订单 的 打印 技术 
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51 开发 背景 














随 着 我 国信 息 化 进程 的 全 面 推进 ， 各 领域 的 信息 化 进程 都 在 飞速 的 发 展 ， 同 样 也 推动 着 物流 领域 
的 信息 化 进程 飞快 的 向 前 发 展 。 由 于 信息 化 进程 的 全 面 推进 ， 对 现代 物流 提出 更 高 的 要 求 : 信息 化 、 
自动 化 、 网 络 化 、 智 能 化 和 柔性 化 等 。 物 流行 业 的 竞争 日 益 激烈 ， 客 户 需求 的 标准 也 越 来 越 高 ， 物 流 
企业 要 想 在 市 场 中 占有 一 席 之 地 ， 必 须要 建立 一 个 高 效 、 快 捷 、 方 便 的 物流 信息 系统 ， 为 客户 提供 一 
流 的 服务 ， 并 且 能 够 及 时 、 准 确 地 掌握 客户 的 需求 ， 对 客户 的 需求 做 出 快速 的 反映 ， 在 最 短 的 时 间 内 
以 最 大 限度 挖掘 和 优化 物流 资源 来 满足 客户 的 需求 ， 从 而 建立 高 效 的 数字 化 物流 经 济 。 


5.2 系统 分 析 


5.2.1 需求 分 析 


随 着 现代 物流 信息 化 进程 的 加 快 ， 传 统 的 物流 管理 方式 已 经 不 再 适应 当前 物流 发 展 的 要 求 ， 取 而 
代 之 的 将 是 以 计算 机 为 基础 的 网 络 化 物流 管理 方式 。 某 物流 配送 公司 为 适应 物流 信息 化 进程 的 发 展 ， 
急需 开发 一 个 物流 配送 系统 ， 通 过 网 络 来 实现 对 物流 操作 流程 进行 管理 ， 不 但 可 以 为 企业 的 运营 过 程 
节省 大 量 的 人 力 、 物 力 、 财 力 和 时 间 ， 提 高 物流 系统 运行 的 效率 ， 而 且 可 以 使 企业 在 客户 心中 树立 一 
个 全 新 的 形象 ， 为 企业 的 发 展商 定 一 个 良好 的 基础 。 现 根据 对 该 物流 配送 公司 的 实际 调查 ， 以 及 公司 
的 具体 要 求 ， 制 定 出 物流 配送 信息 网 的 规划 方案 。 具 体内 容 如 下 
网 站 页 面 设计 要 求 美观 大 方 ， 能 够 展示 企业 形象 。 
为 企业 在 客户 中 树立 一 个 全 新 的 形象 。 
网 站 的 操作 流程 简单 、 方 便 ， 能 够 提高 工作 效率 。 
提供 物流 配送 的 全 程 跟踪 。 
提供 配送 信息 的 及 时 查询 。 
实现 对 配送 车 辆 的 管理 。 
实现 对 客户 信息 的 管理 。 
实现 发 货 单 打印 的 功能 。 


5.2.2 可 行 性 分 析 

物流 配送 信息 网 的 开发 不 但 能 使 物流 企业 走 上 科学 化 、 网 络 化 管理 的 道路 ， 而 且 能 够 为 企业 带 来 
巨大 的 经 济 效益 和 技术 上 飞速 的 发 展 。 

1. 经 济 性 

科学 的 管理 方法 ， 便 捷 的 操作 环境 ， 系 统 的 经 营 模 式 ， 将 为 企业 带 来 更 多 的 客户 资源 ， 树 立 企业 
的 品牌 形象 ， 提 高 企业 的 经 济 效益 。 

2. 技术 性 

网 络 化 的 物流 管理 方式 ， 在 操作 过 程 中 能 够 快捷 地 查找 出 车 源 信息 、 客 户 订单 以 及 客户 信息 ， 能 
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够 对 货物 进行 全 程 跟踪 ， 了 解 货物 的 托运 情况 ， 从 而 使 企业 能 够 根据 实际 情况 ， 做 好 运营 过 程 中 的 各 
项 准备 工作 ， 作 出 及 时 准确 的 调整 :能够 保证 托运 人 以 及 收 货 人 对 货物 进行 及 时 的 处 理 。 


5.3 系统 设计 


5.3.1 系统 目标 


结合 目前 网 络 上 物流 配送 系统 的 设计 方案 ， 对 客户 做 的 调查 结果 以 及 企业 的 实际 需求 ， 本 项 目 在 
设计 时 应 该 满足 以 下 目标 ; 
界面 设计 美观 大 方 、 操 作 简单 。 
功能 完善 、 结 构 清晰 。 
能 够 快速 查询 车 源 信息 。 
能 够 准确 填写 订单 。 
能 够 实现 订单 查询 、 打 印 。 
能 够 实现 对 回 单 处 理 。 
能 够 对 车 源 信息 进行 添加 、 修 改 和 删除 。 
能 够 对 客户 信息 进行 管理 。 
能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 
良好 的 数据 库 系统 支持 。 
系统 运行 稳定 ， 具 备 良好 的 防范 措施 。 


5.3.2 ”系统 功能 结构 
结合 需求 分 析 和 系统 目标 中 的 内 容 ， 物 流 配送 信息 网 的 功能 结构 已 经 设计 完成 。 为 了 使 读者 能 够 
更 清楚 地 了 解 网 站 的 结构 ， 下 面 给 出 物流 配送 信息 网 的 功能 模块 结构 图 和 工作 流程 图 。 
物流 配送 信息 网 的 功能 模块 结构 图 如 图 5.1 所 示 。 
回执 单 确认 


查询 车 源 信息 
使 用 该 车 广 


因 办 办 办 办 办 办 国 办 办 办 








添加 客户 
删除 客户 
添加 车 源 信息 















































修改 车 源 信息 






































是 除 车 源 信息 




















图 5.1 物流 配送 信息 网 功能 模块 结构 图 
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物流 配送 信息 网 的 工作 流程 图 如 图 5.2 所 示 。 


mn 
局 

































































客户 
C) 
回执 单 确认 型 车 辆 查询 
加 
管理 员 
已 
人 7 填写 发 货 单 
货物 到 过 站 


5.2 物流 配送 信息 网 的 工作 流程 图 
5.3.3 ”系统 预览 


物流 配送 信息 网 由 多 个 程序 页 面 组 成 ， 下 面 给 出 几 个 典型 页 面 ， 其 他 页 面 参 见 资源 包 中 的 源 程序 。 
物流 配送 信息 网 主页 如 图 5.3 所 示 , 该 页 面 用 于 展示 本 系统 的 车 源 信息 查询 模块 登录 页 面 如 图 5.4 所 示 。 


物流 配送 信息 网 时 es 





1 月 15E 过 达 该 车 





CPS 07500273e 
Cepyngm@2017 二 村 各 日 和 技 际 公司 











图 5.3 主页 (资源 包 \TM\05\Logistics\index.php) 

发 货 单 管理 页 面 如 图 5.5 所 示 ， 该 页 面 主要 用 于 实现 对 发 货 单 的 处 理 。 单 击 “ 新 增 ” 按 钮 ， 可 以 添 
加 发 货 单 。 单 击 “ 回 执 确认 ”按钮 ， 可 以 确认 发 货 单 回执 。 单 击 “ 查 看 明细 ”按钮 ， 可 以 查看 发 货 单 
明细 。 单 击 “ 编 辑 ” 按 钮 ， 可 以 编辑 发 货 单 。 单 击 “ 删 除 ” 按 钮 ， 可 以 删除 发 货 单 。 在 输入 框 内 填写 
“发 货 单 号 ”或 “发 货 人 ”， 单 击 “ 搜 索 ” 图 标 ， 即 可 搜索 满足 搜索 条 件 的 发 货 单 。 
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后 台 登 录 























物流 配送 信息 网 时 四 8 员 
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物流 配送 信息 网 








5.5 发 货 单 管理 页 面 〈 资 源 包 \TMWS\Logisticsvorderphp) 


客户 信息 管理 模块 的 页 面 效 果 如 图 5.6 所 示 ， 该 页 面 实现 对 客户 信息 的 添加 


、 修 改 和 删除 操作 。 
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车 源 信息 管理 模块 的 页 面 效 果 如 图 5.7 所 示 ， 该 页 面 实现 对 车 源 信息 的 添加 


图 5.6 客户 信息 管理 页 面 (资源 包 \TM\05\Logistics\customer.php) 


、 修 改 和 删除 操作 。 
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物流 配送 信息 网 下 Ee 
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长 要 
市 
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EE 
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图 5.7 车 源 信息 管理 页 面 (资源 包 \TM\05\Logistics\carphp) 
修改 管理 员 密 码 模块 的 页 面 效 果 如 图 5.8 和 图 5.9 所 示 ， 该 页 面 实现 对 管理 员 密 码 的 修改 。 














物流 配送 信息 网 
明科 入 2 

新 安 码 

四 
图 5.8 单 击 修改 密码 页 面 5.9 ”修改 密码 页 面 (资源 包 \TM\05\Logistics\change_pwd.php) 

5.3.4 开发 环境 

在 开发 物流 配送 信息 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 ，Windows/Linux (推荐 )。 
服务 器 : Apache 2.4.18。 

PHP 软件 : PHP 7.0。 

数据 库 : MySQL 5.5.47。 

MySQL 图 形 化 管理 软件 : Navicat 11。 
开发 工具 : PhpStorm 9。 

浏览 器 : 谷歌 浏览 器 。 


I 
办 办 办 办 办 办 加 
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2. 客户 端 
回 ”浏览 器 : 谷歌 浏览 器 。 
5.3.5 ”文件 夹 组 织 结构 


在 进行 网 站 开发 之 前 ， 要 对 网 站 的 整体 文件 夹 组 织 架 ei i 
构 进行 规划 。 对 网 站 中 使 用 的 文件 进行 合理 的 分 类 , 分 别 放 r 口 Conn 一 一 一 数据 库 连 接 文件 


置 于 不 同 的 文件 夹 下 。 通过 对 文件 夹 组 织 架构 的 规划 , 可 以 “i 
确保 网 站 文件 目录 明确 、 条 理 清晰 , 同样 也 便于 网 站 后 期 的 * DVew 公共 视图 文件 


更 新 和 维护 。 物 流 配 送信 息 网 的 文件 夹 组 织 结构 如 图 5.10 


3 图 5.10 文件 夹 组 织 结构 
所 示 。 


5.4 ”数据 库 设计 





物流 配送 信息 网 必须 拥有 数据 库 的 支持 ， 所 有 物流 配送 的 数据 都 应 该 存储 到 数据 。 “ 
库 中 ， 便 于 管理 员 查 找 车 辆 、 订 单 和 客户 的 信息 。 如 果 没 有 数据 库 的 支持 ， 那 么 物流 配送 信息 网 将 没 
有 任何 意义 。 本 节 将 对 物流 配送 信息 网 的 数据 库 设 计 进 行 详细 介绍 。 


5.4.1 数据 库 分 析 


物流 配送 信息 网 是 一 个 中 小 型 的 企业 管理 系统 ， 考 虑 到 开发 的 成 本 、 搭 配 的 合理 性 以 及 操作 的 灵 
活性 等 ， 使 用 MySQL 数据 库 是 最 佳 的 选择 。MySQL 数据 库 是 完全 免费 的 ， 使 用 它 不 需要 任何 费用 ， 
可 以 直接 从 网 上 免费 下 载 : MySQL 数据 库 的 操作 也 非常 方便 ， 不 但 可 以 在 命令 模式 下 操作 ， 而 且 配备 
目前 比较 流行 的 图 形 化 管理 工具 Navicat， 能 够 轻松 地 实现 对 MySQL 数据 库 的 管理 和 操作 。 


5.4.2 ”数据库 概念 设计 
根据 上 述 各 节 对 物流 配送 信息 网 做 的 需求 分 析 和 系统 设计 ,整理 出 物流 配送 信息 网 的 实体 关系 E-R 
图 。 其 中 包括 管理 员 信息 实体 、 车 源 信息 实 体 、 车 辆 日 志 信息 实体 、 客 户 信息 实体 和 发 货 单 信息 实体 。 
1. 管理 员 信息 实体 


管理 员 信息 实体 用 于 存储 管理 员 的 登录 名 称 和 密码 信息 ， 包 括 用 户 名 和 密码 两 项 内 容 。 管 理 员 信 
息 实体 的 E-R 图 如 图 5.11 所 示 。 


2。 车 源 信息 实体 用 户 名 管理 员 信息 密码 
车 源 信息 实体 用 于 存储 企业 拥有 的 车 辆 信息 ， 图 5.11 管理 员 信息 实体 的 E-R 图 


包括 车 主 姓 名 、 车 主 身份 证 号 、 车 牌号 码 、 车 主 联 
系 电话 、 车 主 家 庭 地 址 、 车 辆 行驶 路 线 和 车 辆 描述 。 车 源 信息 实体 的 E-R 图 如 图 5.12 所 示 。 
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皇家 放 地 二》 。 CC 车 让 姓名》 ET 
CC 车 辆 行驶 路 线 


车 主 身份 证 号 车 源 信息 
车 主 联 系 电话 车 辆 描述 


图 5.12 车 源 信息 实体 的 E-R 图 




















3 车辆 日 志 信息 实体 


车 辆 日 志 信 息 实体 记录 车 辆 当前 是 否 被 占用 ， 以 及 使 用 的 时 间 、 执 行 的 任务 ， 包 括 车 牌号 码 、 车 
辆 日 志 信息 (详细 的 车 辆 使 用 描述 )、 日 志 时 间 和 发 货 单 ID( 即 执行 的 任务 )。 车辆 日 志 信息 实体 的 E-R 
图 如 图 5.13 所 示 。 

4. 客户 信息 实体 


客户 信息 实体 用 于 存储 客户 的 信息 ， 包 括 客户 姓名 、 客 户 电话 和 客户 联系 地 址 。 客 户 信息 实体 的 
E-R 图 如 图 5.14 所 示 。 


车 辆 日 志 
































图 5.13 车辆 日 志 信息 实体 的 E-R 图 图 5.14 客户 信息 实体 的 E-R 图 
5、 发 货 单 信息 实体 
发 货 单 信息 实体 用 于 存储 客户 填写 
的 发 货 单 信息 ， 包 括 车 牌号 码 、 车 主 电 CC > CC WA 


话 、 货 物 描述 、 发 货 人 、 发 货 时 间 、 发 


货 人 电话 、 发 货 地 址 、 收 货 人 、 收 货 人 


电话 、 收 货 地 址 和 付款 方式 。 发 货 单 售 。 C 二 电话 发 代 音信 息 
息 实体 的 E-R 图 如 图 5.15 所 示 。 
5.4.3 ”创建 数据 库 及 数据 表 图 5.15 发 货 单 信息 实体 的 ER 图 

















在 物流 配送 信息 网 中 应 用 的 是 db_logistics 数据 库 ， 其 中 涉及 5 个 数据 表 ， 如 表 5.1 所 示 。 
表 5.1 数据 表 名 称 及 说 明 






























说 明 表 名 说 有 明 
tb_admin | 管理 员 信息 表 tb _customer 客户 信息 表 
tb car 车 源 信息 表 tb shopping | 发 货 单 信息 表 












车 辆 日 志 信息 表 
下 面 介绍 物流 配送 信息 网 中 使 用 的 数据 表 。 
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1. tb_admin (管理 员 信息 表 ) 
管理 员 信息 表 主 要 用 于 存储 管理 员 的 登录 名 和 密码 。 该 数据 表 的 结构 如 表 5.2 所 示 。 
表 5.2 管理 员 信 息 表 结 构 


备注 
int(10) NO 是 
varchar(50) 管理 员 名 
| wa | | x | 密友 
2. tb_car (车 源 信息 表 ) 
车 源 信息 表 主 要 用 于 存储 配送 公司 拥有 的 车 辆 信息 ， 主 要 包括 车 主 姓名 、 车 主 的 身份 证 号 码 、 车 
牌号 码 、 车 主 电话 、 车 主 联 系 地 址 、 车 辆 行驶 路 线 和 车 辆 描述 。 该 数据 表 的 结构 如 表 5.3 所 示 。 


表 5.3 车 源 信息 表 结构 








字段 名 数据 类 型 自动 递增 备注 
id my) | | xo | 是 主键 
username varchar(50) | | 车 主 名 称 
User number varchar5) | | 身份 证 号 
car_ number varchar5) | | 车 牌号 
tl vba | | so | 车 主 电话 
address varchar(80) | | 车 主 地 址 
car road mediumtext | | 路 线 
car_content mediumtext | | 车 辆 描述 








3. tb_car_log( 车 辆 日 志 信 息 表 ) 


车 辆 日 志 信 息 表 主 要 用 于 存储 车 辆 的 使 用 情况 信息 ， 主 要 包括 车 牌号 码 、 车 辆 日 志 信息 、 日 志 创 
建 时 间 和 发 货 单 的 ID。 该 数据 表 的 结构 如 表 5.4 所 示 。 


表 5.4 车 源 信息 表 结 构 










数据 类 型 














log id | inao 是 | 主键 
car_number | varchar(50) | 车 牌号 
car log | mediumtext | 车 辆 日 志 信 息 
log date | uatetime | 日 志 创 建 时 间 








varchar(50) 发 货 单 ID 





fahuo id 
4. tb_customer (客户 信息 表 ) 
客户 信息 表 主 要 用 于 存储 客户 的 信息 ， 包 括 客户 姓名 、 客 户 电话 以 及 联系 地 址 。 该 数据 表 的 结构 
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如 表 5.5 所 示 。 


表 5.5 客户 信息 表 结 构 

















字 段 名 数据 类 型 
customer id | int(10) 是 | 主键 
Customer user | varchar(50) | 客户 姓名 









varchar(50) 
varchar(80) 


customer tel 








customer address 





5. tb_shopping (发 货 单 信息 表 ) 


发 货 单 信息 表 主 要 用 于 存储 客户 填写 的 发 货 单 中 的 信息 ， 主 要 包括 车 牌号 码 、 车 主 电话 、 货 物 描 
述 、 发 货 人 、 发 货 时 间 、 发 货 人 电话 、 发 货 地 址 、 收 货 人 、 收 货 人 电话 、 收 货 地 址 和 付款 方式 。 该 数 
据 表 的 结构 如 表 5.6 所 示 。 


表 5.6 发 货 单 信息 表 结构 


字段 名 | 数据 类 型 | 默认 值 | 允许 为 空 | 自动 递增 | 备注 
ia indg) El 本- 面 主键 

car number varchar(50) | No | | 车 牌号 

fahuo content | mediumtext | NO | | 货物 描述 

fahuo id varchar(50) | No | | 发 货 单 D 

fahuo_user varchar(50) | No | | 发 人 人 

fahuo_time datetime | wo | | 发 货 时 间 


fahuo ys tinyint(20) 0 | YEs | | 回执 单 确认 ，0 未 确认 ，1; 已 确认 
fahuo 人 tinyint(20) 0 | NO | | 付款 方式 0: 发 货 人 付款 ，1: 第 三 方 付款 





fahuo_tel varchar(50) | wo | | 发 货 人 电话 
fahuo_address mediumtext | No | | 发 货 地 址 
car tel varchar(50) [人 一 一 车 主 电 话 
shouhuo_user varchar(50) NO 收 货 人 姓名 
shouhuo_address | mediumtext | wo | | 收 货 人 地 址 














shouhuo tel varchar(50) | NO | | 收 货 人 电话 











5.5 网 站 首页 设计 





i 视 
5.5.1 网 站 首页 概述 

本 系统 是 专 为 管理 员 开 发 的 后 台 系统 ， 而 网 站 首页 是 浏览 者 看 到 的 第 一 视觉 界面 ， 所 以 在 设计 网 
站 的 首页 时 应 该 将 网 站 中 主要 的 内 容 尽量 展示 给 浏览 者 ， 让 浏览 者 能 够 更 快 地 了 解 网 站 的 内 容 。 物 流 
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配送 信息 网 的 首页 主要 包括 如 下 功能 。 

加 “车 源 信息 查 询 : 主要 实现 车 辆 信息 查询 ， 及 时 为 客户 选择 货物 配送 的 方案 。 

回 ”发 货 单 管理 : 主要 用 于 客户 发 货 单 信息 填写 。 

回 客户 信息 管理 ; 主要 用 于 对 客户 的 信息 进行 管理 。 

回 “车 源 信息 管 理 ; 主要 用 于 对 企业 拥有 的 车 辆 信息 进行 管理 。 

回 ”修改 密码 : 主要 用 于 对 管理 员 登 录 的 密码 进行 修改 。 

本 案例 中 提供 的 网 站 首页 如 图 5.16 所 示 。 该 页 面 在 本 书 资源 包 中 的 路 径 为 \TM\05\Logistics\ 
index.php 。 





到 樟 兴 内 容 展 不 i 底部 信息 


图 5.16 物流 配送 信息 网 首页 


5.5.2 ”网 站 首页 技术 分 析 


物流 配送 信息 网 是 为 物流 配送 公司 设计 的 一 个 物流 管理 系统 ， 其 页 面 的 设计 突出 简洁 、 方 便 ， 功 
能 的 实现 以 便于 操作 和 维护 为 根本 。 网 站 首页 的 页 面 设计 如 图 5.16 所 示 ， 由 一 个 简单 的 框架 组 成 ， 其 
结构 由 如 下 4 个 部 分 组 成 。 

回 ”顶部 信息 : 主要 用 于 展示 网 站 logo， 以 及 显示 “退出 ”和 “修改 密码 ”的 按钮 。 

回 ”菜单 栏 : 主要 展示 网 站 的 主要 功能 菜单 。 

回 ”功能 模块 内 容 : 主要 显示 对 应 菜单 的 功能 。 

回 ”底部 信息 : 主要 用 于 展示 网 站 版 权 信息 。 

在 以 上 4 个 部 分 中 ， 顶 部 信息 、 菜 单 栏 和 底部 信息 均 是 固定 不 变 的 内 容 ， 可 以 作为 通用 模板 在 每 
个 页 面 中 使 用 。 主 体 框架 结构 代码 如 下 : 

<!-- 头 部 内 容 --> 


<?php include('View/header.html') ?> 
<div class="container-fluid"> 
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<!-- 顶 部 导航 --> 
<?php include("View/nav.html') ?> 
<!- 主 区 域 开 始 --> 
<div class="row" style="margin-top:70px"> 
<!-- 左 侧 菜单 --> 
<?php include(Viewmenu.html') ?> 
<!- 右 侧 主 区 域 开始 --> 
<div class="main-right col-md-10 col-md-offset-2"> 
<div class="col-md-12"> 
<!- 省 略 部 分 代码 -> 
</div> 
<!-- 右 侧 主 区 域 结束 --> 
</div> 
<!- 主 区 域 结 束 --> 
</div> 
<!-- 底 部 信息 --> 
<?php include('View\footer.html") ?> 





选择 不 同 的 菜单 栏 ， 会 展示 相应 的 功能 模块 内 容 。 在 默认 情况 下 ， 用 户 成 功 登 录 后 进入 到 车 源 信 
息 查询 模块 ， 因 为 该 模块 最 常用 ， 所 以 在 index.php 页 面 中 显示 该 模块 。 


5.5.3 ”网 站 首页 的 实现 过 程 


物流 配送 信息 网 的 首页 设计 非常 简单 ， 主 要 应 用 一 个 简单 的 菜单 功能 ， 并 且 以 超 链接 作为 辅助 ， 
最 终 实现 在 不 同 功能 模块 之 间 的 跳 转 。 

首先 连接 数据 库 ， 判 断 登 录用 户 的 身份 ， 如 果 是 管理 员 ， 则 可 以 进入 网 站 进行 操作 ， 否 则 将 提示 
“请 先 登 录 !” 并 且 跳 转 到 网 站 的 登录 页 面 。 关 键 代 码 如 下 : 


倒 程 01 代码 位 置 ， 资源 包 \TM\05\Logistics\index.php 


<?php 
include('Conn\config.php’); // 引 入 配置 文件 
include('Library\function.php'); /引入 函数 库 
/独断 是 否 登录 
if(IcheckLogin()}{ 


msg(2," 请 先 登录 ',login.php'); 。 // 如 未 登录 时 跳 转 到 登录 页 面 


} 
< 上 -省 略 了 部 分 代码 --> 


2 


上 述 代 码 中 使 用 的 checkLogin0 是 一 个 自 定义 函数 , 用 于 判读 用 户 是 否 登 录 。 本 项 目 中 所 有 自 定义 
函数 均 在 “/Library/function php” 文 件 中 。checkLogin0) 函 数 的 代码 如 下 : 


倒 程 02 ”代码 位 置 : 资源 包 \TM\05\Logistics\Libary\function.php 





<?php 
function checkLogin() 


全 
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session_start(); /开启 session 
/用 户 未 登录 返回 false， 已 登录 返回 true 
iflisset($_SESSIONTuser]) || empty($_SESSIONruser]) 
{ 

return false; 
. 
return true; 


} 
?> 


在 未 登录 的 情况 下 访问 index.php 文件 , 会 执行 msg(2,' 请 先 登录 ','login.php") 语 句 , 其 中 msg0 函 数 
也 是 一 个 自 定义 的 函数 ， 用 来 实现 提示 信息 并 跳 转 页 面 的 功能 。 具 体 代码 如 下 : 


倒 程 03 代码 位 置 ， 资源 包 \TM\05\Logistics\Library\function.php 
pe 
* 消息 提示 
* @param int $type 1: 成 功 2: 失 败 
* @param null $msg 
* @param null $url 
和 
function msg($type, $msg = null, $url = null) 
{ 





S$toUrl = "Location:msg.php?type={$type}"; // 跳 转 到 msg.php 函数 ， 并 且 传递 参数 
// 当 msg 为 空 时 url 不 写 入 


StoUrl .= $msg ? "&msg={$msg}" : "; // 将 $msg 参数 拼接 到 url 中 
// 当 url 为 空 toUrl 不 写 入 

S$toUrl .= $url ? "&url={$url}" : "; // 将 $url 参数 拼接 到 url 中 
header($toUrl); 

exit; 


在 未 登录 的 情况 下 访问 localhost/logistics/index.php 文件 ， 运 行 效果 如 图 5.17 所 示 。 
| 


| € 3 © |© ocalhoslogistics/msgphp?typ 











图 5.17 跳 转 到 登录 页 面 
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5.6 车 源 信息 查询 模块 设计 





频 讲 解 


5.6.1 车 源 信息 查询 模块 概述 


车 源 信息 查询 模块 的 功能 是 根据 容 户 提供 的 货 。 加 
物 配 送 的 地 点, 从 数据 库 中 查询 有 关 该 路 线 的 车 辆 ， 一 让 一 < 局 
为 客户 提供 一 个 合理 配送 的 路 线 ， 最 后 确定 发 货 订 


























日 
























































单 。 操 作 流程 如 图 5.18 所 示 。 i i 

5.6.2 车 源 信息 查询 模块 技术 分 析 
车 源 信息 查询 模块 的 主要 功能 就 是 查询 车 辆 的 。 [WR 证 车 辆 查询 

使 用 信息 ， 为 客户 提供 最 合适 的 货物 配送 路 线 。 其 

中 应 用 的 关键 技术 自然 就 是 查询 方法 ， 为 了 给 客户 图 5.18 车 源 信息 查询 的 操作 流程 

提供 最 合适 、 最 满意 、 最 快捷 的 配送 服务 ， 这 里 使 

用 的 是 模糊 查询 技术 。 


通过 模糊 查询 技术 ， 只 要 客户 提出 配送 货物 的 出 发 地 点 和 到 达 地 点 ， 管 理 员 就 可 以 从 数据 库 中 提 
取出 所 有 与 该 路 线 相关 的 车 辆 信息 ， 包 括 车 辆 的 承载 能 力 、 车 辆 是 否 被 占用 和 车 辆 的 使 用 情况 ， 客 户 
可 以 根据 实际 的 情况 〈 时 间 、 货 物 数 量 、 路 线 等 ) 选择 车 辆 ， 确 认 后 填写 发 货 单 。 

在 MySQL 中 ， 可 以 使 用 like 语句 实现 模糊 查询 ， 语 法 格式 如 下 : 


SELECT field1, field2,..fieldN “FROM table name WHERE field1 LIKE condition1 [AND [OR] filed2 = 
"Somevalue' 


like 子 句 可 以 代替 等 号 ， 通 常 与 % 一 同 使 用 ， 类 似 于 一 个 元 字符 的 搜索 。 例 如 ， 查 找 所 有 以 “ 王 ” 
字 开 头 的 用 户 ， 查 询 语句 如 下 : 


select * from tb_car where username like " 王 %" 








再 如 ， 查 找 所 有 名 字 中 包含 “ 明 ” 的 用 户 ， 查 询 语句 如 下 : 


select * from tb_car where usemame like "% 了 明 %" 
5.6.3 ”车 源 信息 查询 模块 的 实现 过 程 
国 车 源 信息 查询 模块 使 用 的 数据 表 : tb car、tb car log 


模糊 查询 技术 的 实现 是 在 index.php 文件 中 完成 的 , 首先 要 与 数据 库 建立 连接 ; 然后 创建 一 个 form 
表单 ， 设 置 两 个 文本 框 start 和 end 用 于 提交 出 发 地 点 和 到 达 地 点 ， 将 表单 的 值 提 交 到 当前 页 ， 最 后 接 
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收 数据 ， 以 表单 中 获取 的 数据 为 条 件 ， 执 行 like 模糊 查询 ， 从 数据 库 的 指定 表 中 查询 符合 条 件 的 信息 ， 
并 且 将 查询 出 的 信息 显示 到 页 面 中 。 程 序 的 关键 代码 如 下 : 


倒 程 04 代码 位 置 资源 包 \TM\05\Logistics\index.php 


<?php 
include('Conn\config.php’); /1/ 引 入 配置 文件 
include('Library\function.php’); /11 引入 函数 库 
1/ 判断 是 否 登录 
if(IcheckLogin()){ 


msg(2," 请 先 登录 ',login.php'); 


} 
// 获 取 查 询 条件 

@ S$start=isset($_GET['start])? $_GET[start] :™; 
$end =isset($_GET[end])? $_GET[end] : ""; 


/* 分 页 设置 */ 
@ $page =isset($_GET[page]) ? intval($_GET[page]) : 1; /检查 page 参数 
$page = max($page, 1); /把 page 与 1 对 比 ， 取 中 间 最 大 值 
$pageSize = 5; /每 页 显示 条 数 
$show = 6; /按钮 显示 数量 
Soffset = ($page - 1) * $pageSize; // 设 置 起 点 
/获取 分 页 数据 


© $query_all = "select count(*) from tb_car where car_road like '%$start%' and car_road like '%$end%' "; 
Sresult = $pdo->prepare($query_all); 
$result->execute(); 
Stotal = $result->fetchColumn(); 
®@ S$pages= pages($total,$page,$pageSize,$show); // 调 用 分 页 方法 
/* 筛选 数据 */ 
© $query = "select tb_car*tb_car_log.car_log from tb_car left join tb_car_log on tb_car log.car_number = 
tb_car.car_number "; 
$query .= "where car_road like '%$start%' and car_road like '%$end%' order by id desc limit 
{Soffset},{$pageSize} ", 
Sres = $pdo->prepare($query); 
Sres->execute(); 
?> 


人 代码 贴 十 

@ start: 指定 出 发 地 点 ， 当 没有 输入 出 发 地 点 时 ， 默 认 传 递 空 值 ， 会 查询 出 所 有 数据 。 

@ page: 获取 当前 所 在 页 ， 默 认为 第 1 页。 

@@ like: 执行 模糊 查询 ， 从 数据 库 中 读 取 符合 条 件 的 数据 总 数 ， 为 分 页 做 准备 。 

@ pages0: 自 定义 分 页 函数 ， 接 收 4 个 参数 ， 返 回 分 页 样式 表 。 

@ join: 使 用 join 实现 级 联 查询 ， 因 为 在 获取 也 _car 表 所 有 字段 信息 的 同时 ， 还 需要 获取 tb car log 表 的 tb_car log 
字段 信息 ， 所 以 使 用 left join 左 级 联 方式 联合 查询 。 


在 车 源 信息 查询 模块 的 查询 输入 框 中 ， 出 发 地 点 输入 “长 春 ”， 目 的 地 输入 “北京 ”， 单 击 “ 搜 
索 ” 图 标 ， 查 询 所 有 满足 条 件 的 车 源 ， 运 行 效果 如 图 5.19 所 示 。 
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es 可 去 
车 牌号 三 路 法 车 寅 后 述 使 用 日 志 是 否 使 用 


吉 A78747 长 春 北 京 IN 比 了 让 T7 4x2 怠 工 50mm 切 各 车 人 1 月 15 日 庆 达 预 J 区 < 


图 5.19 查询 车 源 信息 


5.7 发 货 单 管理 模块 设计 








5.7.1 发 货 单 管理 模块 概述 


发 货 单 管理 模块 主要 包括 发 货 单 的 填写 、 发 货 单 查询 、 发 货 单打 印 和 发 货 单 删除 。 发 货 单 管理 模 
块 的 主要 功能 如 图 5.20 所 示 。 





管理 员 


图 5.20 发 货 单 管理 模块 的 功能 结构 图 
5.7.2 发 货 单 管理 模块 技术 分 析 


如 何 才能 发 挥 函 数 的 最 大 作用 ? 设计 者 不 但 要 能 够 从 表面 理解 函数 的 功能 和 作用 ， 而 且 要 能 够 以 
一 种 发 散 的 思维 方式 去 考虑 问题 ， 不 要 被 其 本 身 的 功能 所 束缚 ， 要 将 函数 中 某 些 功能 有 机 地 结合 起 来 ， 
从 而 发 挥 其 更 大 的 作用 。 

在 发 货 单 管理 模块 中 应 用 到 一 个 发 货 单 的 编号 ， 该 编号 是 发 货 单 的 唯一 标识 ， 不 允许 存在 重复 。 
在 编写 生成 这 个 编号 的 程序 时 ， 首 选 方案 是 应 用 随机 函数 获取 随机 的 数字 作为 编号 ， 这 是 一 个 最 直接 
的 做 法 ， 但 是 考虑 到 发 货 单 编号 不 但 具有 唯一 性 ， 而 且 还 要 使 其 具有 一 定 的 标志 性 ， 从 这 个 角度 考虑 
使 用 随机 函数 就 有 些 不 适合 ， 因 为 其 不 具备 一 定 的 标志 性 。 这 时 ， 如 果 应 用 date0 函 数 ， 则 可 以 达到 上 
述 所 说 的 要 求 ， 不 但 编号 具有 唯一 性 ， 而 且 还 有 规律 可 循 。 

为 了 能 够 更 好 地 理解 这 种 发 散 思 维 的 编程 思想 ， 首 先 介绍 一 下 date0 函 数 。 语 法 格式 如 下 ;: 


string date(string format, int timestamp) 


该 函数 返回 将 参数 timestamp 按照 指定 格式 格式 化 而 产生 的 字符 串 , 其 中 参数 timestamp 是 可 选 的 ， 
默认 值 为 tme0， 即 如 果 没 有 给 出 时 间 戳 ， 则 使 用 本 地 当前 时 间 。 
date() 函 数 的 参数 format 的 格式 化 选项 如 表 5.7 所 示 。 


局 
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表 5.7 参数 format 的 格式 化 选项 
说 明 
小 写 的 上 午 和 下 午 值 ， 返 回 值 为 am 或 pm 
大 写 的 上 午 和 下 午 值 ， 返 回 值 为 AM 或 PM 
Swatch Intemet 标准 时 ， 返 回 值 为 000 一 999 
月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 为 01 一 31 
星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 为 Mon 一 Sun 
月 份 ， 完 整 的 文本 格式 ， 返 回 值 为 January 一 December 
小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 
小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 0 一 23 
小 时 ，12 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 01 一 12 
小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 00 一 23 
有 前 导 零 的 分 钟 数 ， 返 回 值 为 00 一 59 
判断 是 否 为 夏令 时 ， 如 果 是 ， 返 回 值 为 1， 否则 为 0 
月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 为 1 一 31 
星期 数 ， 完 整 的 文本 格式 ， 返 回 值 为 Sunday 到 Saturday 
判断 是 否 为 六 年 ， 如 果 是 ， 返 回 值 为 1， 否则 为 0 
数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 为 01 一 12 
3 个 字母 缩写 表示 的 月 份 ， 返 回 值 为 Jan 一 Dec 
数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 
与 格林 威 治 时 间 相差 的 小 时 数 ， 例 如 0200 
RFC 822 格式 的 日 期 ,例如 Thu，21 Dec 2000 16: 01: 07 +0200 
秒 数 ， 有 前 导 零 ， 返 回 值 为 00 一 59 
每 月 天 数 后 面 的 英文 后 级 ，2 个 字符 ， 例 如 st、nd、rd 或 者 也 。 可 以 和 j 一 起 使 用 
指定 月 份 所 应 有 的 天 数 
本 机 所 在 的 时 区 
从 UNIX 纪元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 
星期 中 的 第 几 天 ， 数 字 表示 ， 返 回 值 为 0 一 6 
ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 
4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 
2 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
年 份 中 的 第 几 天 ， 返 回 值 为 0 一 366 
时 差 偏 移 量 的 秒 数 。UTC 西边 的 时 区 偏 移 量 总 是 负 的 ，UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 
返回 值 为 -43200 一 43200 


在 本 项 目 中 的 发 货 单 填 单 模块 中 就 是 应 用 date0 函 数 获 取 当 前 时 间作 为 发 货 单 的 编号 。 关 键 代码 如 下 : 


倒 程 05 代码 位 置 资源 包 \TM\05\Logistics\add_order.php 


<td> 发 货 单 号 :</td> 
<td> 
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<input type="text" id="fahuo_id" class="form-control" name="fahuo_id" readonly 
value="<?php echo date("YmdHis").$car_id;?>"> 
</td> 





其 运行 结果 如 图 5.21 所 示 。 


二 号 





生成 的 订单 号 


HE 人 家 Am 到 


图 5.21 发 货 单 编号 生成 的 运行 结果 
5.7.3 发 货 单 填 单 的 实现 过 程 


回 发 货 单 填 单 模块 使 用 的 数据 表 : tb_customer、tb_car log、tb_shopping 

发 货 单 的 填写 是 客户 在 确定 物流 配送 路 线 以 后 填写 的 一 个 货物 配送 详细 信息 单据 ， 内 容 包括 发 货 
单 编 号 、 车 牌号 码 、 车 主 电话 、 发 货 人 、 发 货 人 电话 、 发 货 地 址 、 付 款 方式 、 货 物 描述 、 收 货 人 、 收 
货 人 电话 、 收 货 地 址 和 说 明 。 发 货 单 填 单 的 运行 结果 如 图 5.22 所 示 。 


许 地 址 ead 
要 伐 地 址 ; 置 关 志 才 区 沙坪 现 万 达 广 场 


5.22 发 货 单 填 单 的 运行 结果 
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发 货 单 填 单 的 实现 通过 add_orderphp 文件 来 完成 ,首先 随机 生成 一 个 订单 编号 ,然后 创建 一 个 form 
表单 ， 最 后 将 表单 中 的 数据 提交 到 数据 库 中 。 在 add_orderphp 文件 中 使 用 的 重要 表单 元 素 如 表 5.8 所 示 。 


表 5.8 发 货 单 填 单 中 使 用 的 重要 表单 元 素 

















名 称 | 元 素 类 型 重要 属性 含义 
add-form | form | action="store_order.php" method="post" | 表单 
fahuo id | text | name="fahuo id" readonly value="<?php echo date("YmdHis"):?>" | 发 货 单 编号 
car number | text | name="car number" value="<?php echo $row['car_ number]?>” | 车 牌号 码 
car tel | text | name="tel" value="<?php echo $row['tel]?>" | 车 主 电话 

<option value="0"> 发 货 人 付款 </option> 
fbuo | ot <option value="1"> 第 三 方 付 款 </option> Wd 








car log textarea IOWS="3" name="car log" 说 明 


单 击 “提交 ”按钮 前 ， 需 要 对 于 表单 中 的 数据 进行 验证 ， 例 如 输入 的 用 户 名 是 否 为 空 ， 电 话 号 码 
格式 是 否 正确 等 。 对 于 前 台数 据 的 验证 ， 可 以 使 用 JavaScript 来 实现 ， 关 键 代码 如 下 : 


倒 程 06 代码 位 置 ， 资源 包 \TM\05\Logistics\add_order.php 
<script> 
$(function() { 
/提交 表单 
$(#add-form').submit(function() { 
if(IcheckPhone(' 失 el',' 请 输入 车 主 电 话 ')X{ 
return false; 








. 
if(IcheckField(#fahuo_user',' 请 输入 发 货 人 !)){ 
return false; 


3 
if(IcheckPhone('#fahuo_tel",' 请 输入 发 货 人 电话 ')){ 
return false; 


j} 
if(IcheckField(#fahuo_content',' 请 输入 货物 信息 1) 
return false; 


} 
if(!icheckField('#fahuo_address',' 请 输入 发 货 地 址 ")){ 
return false; 


} 
iflcheckField(#shouhuo_user," 请 输入 收 货 人 站 多 
return false; 


} 
if(IcheckPhone('#shouhuo_tel',' 请 输入 收 货 人 电话 ')){ 
return false; 


} 
if(lcheckField(#shouhuo_address'," 请 输入 收 货 地 址 if 
return false; 


} 
if(!checkField(#car_log',' 请 输入 车 辆 使 用 信息 !)){ 
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return false; 
} 


return true; 


» 


»); 
/检测 字段 是 否 为 空 
function checkField(id,messagej{ 
if($(id).val() == "X{ 
layertips(message, id, {time: 2000, tips: 2}); 
$(id).focus(); 
return false; 
} 
return true; 


} 
/检测 电话 格式 是 否 正确 
function checkPhone(id,message){ 
if(IcheckField(id,message)X{ 
return false; 


} 

var tel = $(id).val(); 

if(!(/(\d{3}-)(\d{8})SI‘(\d{4}-)(d{7))SI (Nd{4}-)(\d{8})$I^(d{11})S/.test(tel))X 
layer.tips(' 电 话 格式 错误 !", id, {time: 2000, tips: 2}); 
S$(id) focus(); 
return false; 


} 


return true; 


} 
</script> 
上 述 代码 中 ， 定 义 了 2 个 函数 ，checkField0 函 数 用 于 检测 输入 的 字段 值 是 否 为 空 ，checkPhoneO 
函数 使 用 正则 表达 式 方式 检测 电话 号 码 格 式 是 否 正 确 。 如 果 不 满足 条 件 ， 使 用 Layer 弹 层 插件 (官网 : 
http:Wlayerlayui.com) 弹出 提示 信息 。 运 行 效果 如 图 5.23 和 图 5.24 所 示 。 





发 兴 单 号 20171130102159 言 As9KE1 左 让 电话 





图 5.23 未 填写 数据 提示 图 5.24 电话 格式 错误 提示 


将 表单 中 提交 的 数据 存储 到 数据 库 是 通过 store php 文件 来 完成 的 ， 在 该 文件 中 首先 连接 数据 库 ， 
然后 通过 preg_match0 函 数 判 断 表单 中 提交 的 电话 号 码 格式 是 否 正 确 , 最 后 将 表单 中 提交 的 数据 存储 到 
tb_shopping 表 中 , 并 且 将 车 辆 使 用 的 说 明 提 交 到 tb_car_log 表 中 , 将 客户 信息 提交 到 tb_customer 表 中 。 
关键 代码 如 下 : 


倒 程 07 代码 位 置 资源 包 \TM\05\Logistics\store_order.php 





<?php 
include('Conn\config.php’); 1/ 引 入 配置 文件 
include('Library\function.php’); 1 引入 函数 库 


ea 
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// 判 断 是 否 登录 
if(IcheckLogin()){ 
msg(2," 请 先 登 录 ',login.php'); 


} 

/接收 数据 */ 

$fahuo_id =$_POST[fahuo_id]; 
$car_number =$_POST[Tcar_number]; 
$car tel ”=$_POSTTtel]: 
$fahuo_user=$_POST[fahuo_user]; 
$fahuo tel =$_POST[fahuo _tel]; 
$fahuo fk =$_POST[fahuo fk]; 
$fahuo_content = $_POST[fahuo_content]; 
S$fahuo_address = $_POST[fahuo_address'; 
S$fahuo_time = date("Y-m-d H:i:s"); 
$shouhuo_user =$_POST[shouhuo_user]; 
$shouhuo_tel ” =$_POST[shouhuo_tel]; 
$shouhuo_address = $_POST['shouhuo_address]; 
$car log =$_POST[car log]; 


/** 后 台 验 证 数据 */ 
if(empty($fahuo_user)) 


msg(2, ' 请 输入 发 货 人 ! 小 
a 
{ msg(2, ' 请 输入 货物 信息 !); 
rn ey 
msg(2, "请 输入 发 货 地 址 !"); 
ee 
msg(2, "请 输入 收 货 人 !); 
odor dey 


msg(2, ' 请 输入 收 货 地 址 !"); 
} 
if(empty($car_log)) 


msg(2, ' 请 输入 车 辆 使 用 信息 !); 


} 
S$rule = "/^(\d{3}-)(\d{8})$I^(d{4}-)(\d{7)SI(d{4}-) Nd{8})sIANd{11))S; 
if(Ipreg_match($rule,$car_tel)) 


msg(2,' 车 主 电 话 格式 错误 ! '); 
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if(Ipreg_match($rule, $fahuo_tel)) 


msg(2,' 发 货 人 电话 格式 错误 ! '); 


} 
if(Ipreg_match($rule,$shouhuo_tel)) 
{ 

msg(2,' 收 货 人 电话 格式 错误 ! '); 
} 
”新 增 或 编辑 情况 */ 
ifisset($_POST[id]) 

/编辑 */ 

/更 改 tb_shopping 表 

Sid = $_POST[id]; 


$shopping_query = "update tb_shopping set car_number='$car_number',car_tel='$car_tel', 
fahuo_user='$fahuo_user',fahuo_tel='$fahuo_tel',fahuo_address='$fahuo_address', 
fahuo_content='$fahuo_content',fahuo_time='$fahuo_time',fahuo_fk="$fahuo_fk', 
shouhuo_user='$shouhuo_user,shouhuo_address='$shouhuo_address',shouhuo_tel= 
'$shouhuo_tel',car_number='$car_number',car_tel='$car_tel",fahuo_id='$fahuo_id', 
fahuo_user='$fahuo_user',fahuo_tel='$fahuo_tel',fahuo_address='$fahuo_address', 
fahuo_content='$fahuo_content',fahuo_time='$fahuo._time',fahuo_fk='$fahuo_fk', 
shouhuo_user='$shouhuo_user',shouhuo_address='$shouhuo_address',shouhuo_tel= 
'$shouhuo_tel' where id = $id"; 

Sres = $pdo->prepare($shopping_query); 

if(!$res->execute()}{ 

msg(2,' 操 作 失 败 '); 

} 

// 更 改 tb_car_log 表 

S$cat_query = "update tb_car_log set car_log='$car_log',car_number='$car_number', 
log_date='$fahuo_time' where fahuo_id='$fahuo_id' "; 

Sres = $pdo->prepare($cat_query); 

if(!I$res->execute()\{ 

msg(2,' 操 作 失 败 '); 
} 


}else{ 


”新 增 **/ 

// 写 入 tb_shopping 表 

$shopping_query = "insert into tb_shopping (car_number,car_tel,fahuo_id,fahuo_user, fahuo_tel, 
fahuo_address,fahuo_content,fahuo_time,fahuo_fk,shouhuo_user, 
shouhuo_address,shouhuo_tel)values('$car_number','$car_tel','$fahuo_id', 
'$fahuo_user','$fahuo_tel','$fahuo_address','$fahuo_content','$fahuo_time'’, 
'$fahuo_fk','$shouhuo_user','$shouhuo_address','$shouhuo_tel")"; 

Sres = $pdo->prepare($shopping_query); 

if(I$res->execute())\{ 

msg(2, 操作 失败 ); 

} 

// 写 入 tb_car_log 表 

$cat_ query = "insert into tb_car_log(car_log,car_number,log_date,fahuo_id) 


values('$car_log','$car_number','$fahuo_time','$fahuo_id')"; 
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Sres = $pdo->prepare($cat_query); 
if(I$res->execute()\{ 
msg(2,' 操 作 失 败 '); 


} 
// 写 入 tb_customer 表 
$customer_query = "insert into tb_customer (customer_user,customer_tel,customer_address) 
values('$fahuo_user','$fahuo_tel','$fahuo_address')"; 
Sres = $pdo->prepare($customer_query); 
Sres->execute(); 
if(I$res->execute()\{ 
msg(2, 操 作 失 败 ); 
} 
msg(1,' 操 作成 功 ','order.php'); 
?> 


在 上 述 代码 中 ， 首 先 接收 表单 传递 过 来 的 数据 ， 然 后 再 次 检验 数据 〈 不 能 完全 相信 前台 提 交 的 数 
据 ) 。 由 于 “新 增 ” 页 面 和 “编辑 ”页 面 提交 的 内 容 相 同 ， 所 以 在 store_order.php 文件 中 可 以 同时 完 
成 。 判 断 如 果 $_POST[id] 存 在 ， 则 是 从 “编辑 ”页 面 提交 的 数据 ， 使 用 update 语句 更 新 tb_shopping 
和 tb_car log 表 。 否则 是 从 “新 增 ” 页面 提交 的 数据 , 使 用 insert 语句 将 提交 信息 分 别 写 入 tb_shopping、 
tb_car log 和 tb_customer 表 。 


5.7.4 发 货 单 查询 的 实现 过 程 


回 发 货 单 查询 模块 使 用 的 数据 表 : tb car log、tb_shopping 

发 货 单 查询 是 为 了 便于 对 发 货 单 进行 查找 以 及 处 理 而 设计 的 一 个 功能 ， 通 过 其 可 以 准确 地 查找 到 
指定 的 发 货 单 ， 并 且 还 设置 一 个 发 货 单 删除 的 功能 ， 可 以 对 已 经 失效 或 者 作废 的 发 货 单 进行 删除 。 其 
实现 的 原理 主要 是 通过 以 发 货 单 编号 为 条 件 的 精确 查找 或 者 以 发 货 人 姓名 为 条 件 的 模糊 查询 ， 然 后 将 
查询 的 结果 输出 到 页 面 中 。 

发 货 单 查询 的 实现 主要 通过 order.php 文件 完成 ， 首 先 与 数据 库 建立 连接 ， 然 后 创建 一 个 Form 表 
单 ， 通 过 该 表单 来 提交 查询 的 条 件 ， 可 以 选择 精确 查找 以 发 货 单 编号 为 条 件 ) 或 者 模糊 查询 (以 发 
货 人 姓名 为 条 件 )， 最 后 根据 Form 表单 中 提交 的 数据 ， 执 行 查询 语句 ， 从 数据 库 中 读 取出 符合 条 件 的 
发 货 单 的 内 容 。 程 序 的 关键 代码 如 下 : 


倒 程 08 代码 位 置 ， 资源 包 \TM\05\Logistics\order.php 


<?php 
include('Conn\config.php’); /引入 配置 文件 
include(Libraryfunction.php'); 1 引入 函数 库 
// 判 断 是 否 登录 
if(IcheckLogin()){ 


msg(2," 请 先 登 录 ',login.php'); 
} 
@ $keyword= isset($_GET[keyword])? $_GET[keyword] :"; 
// 检 查 page 参数 
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$page = isset($_GET[page]) ? intval($_GET[page]) : 1; 


$page = max($page, 1); // 把 page 与 1 对比， 取 中 间 最 大 值 
$pageSize = 5; /| 每 页 显示 条 数 
$show = 6; /按钮 数量 
$offset = ($page - 1) * $pageSize; 
Ar 分 页 so 
$query_all = "select count(*) from tb_shopping 
© where fahuo_id = '$keyword' or fahuo_user like '%$keyword%' "; 


Sresult = $pdo->prepare($query_all); 
Sresult->execute(); 
Stotal = $result->fetchColumn(); 
$pages = pages($total,$page,$pageSize,$show); /调用 分 页 方法 
/筛选 数据 */ 
$query = "select * from tb_shopping where fahuo_id = '$keyword' or fahuo_user 
like '"%$keyword%' order by id desc limit {$offset},{$pageSize} "; 
S$res = $pdo->prepare($query); 
$res->execute(); 
?> 


二 代码 贴 士 

@ keyword: 接收 传递 过 来 的 搜索 内 容 ， 赋 值 给 变量 $keyword。 如 果 没 有 输入 任何 内 容 ， 则 $keyword 为 空 ， 查 询 所 
有 数据 。 

@ or: or 表示 “或 "， 即 两 个 条 件 任意 一 个 满足 即 可 。 
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图 5.26 以 发 货 人 姓名 为 条 件 的 模糊 查询 结果 
5.7.5 发 货 单 删除 的 实现 过 程 


国 发 货 单 删除 使 用 的 数据 表 : tb_car log、tb_shopping 
在 发 货 单 管理 页 面 单 击 “ 删 除 ” 按 钮 ， 弹 出 一 个 提示 框 ， 如 果 单 击 “ 确 认 ” 按 钮 ， 则 执行 删除 操 


& 


第 5 章 物流 配送 信息 网 (Apache+PHP+MySQL 5.5 实现 ) ”毛毛 毛 


作 ， 和 否则 ， 取 消 删除 。 通 过 JavaScript 和 Layer 弹 层 插件 可 以 实现 该 功能 ， 程 序 的 关键 代码 如 下 : 
倒 程 09 ”代码 位 置 : 资源 包 \TM\05\Logistics\order.php 


/删除 发 货 单 
S$(.del).on(click ,function(){ // 单 击 事件 
var url = $(this).attr('href); // 获 取 跳 转 的 url 
layer.confirm(' 确 认 删 除 该 发 货 单 吗 ?',function(X // 调 用 Layer 弹 层 插件 的 confirm() 方 法 
windowlocation = url; /| 单 击 “ 确 定 ”按钮 后 ， 页 面 跳 转 
入 
return false; /终止 程序 
D); 
页 面 运 行 效果 如 图 5.27 所 示 。 





图 5.27 删除 提示 确认 效果 


发 货 单 删除 的 操作 通过 delete_order.php 文件 来 完成 , 主要 根据 超 链接 中 提供 的 发 货 单 ID 执行 删除 
的 操作 ， 删 除 发 货 单 的 同时 ， 还 要 删除 车 辆 日 志 信息 。 关 键 代 码 如 下 : 


倒 程 10 ”代码 位 置 : 资源 包 \TM\05\Logistics\delete_order.php 


<?php 
include('Conn\config.php’); /引入 配置 文件 
include('Library\function.php'); /引入 函数 库 
// 判 断 是 否 登录 
if(IcheckLogin()){ 


msg(2,' 请 先 登录 ',login.php'); 
} 
S$id = intval($_GET[id]); 
if(empty($id)X{ 
msg(2,' 非 法 操作 ','order.php'); 


/查找 发 货 id 

$query = 'select fahuo_id from tb_shopping where id = ".$id; 
S$res = $pdo->prepare($query); 

Sres->execute(); 

S$fahuo_id = $res->fetchColumn(); 
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?> 


/删除 tb_car_log 表 
$query_log = 'delete from tb_car_log where fahuo_id = :fahuo_id'; 
$res_log = $pdo->prepare($query_log); 
$res_log->bindParam('fahuo_id',$fahuo_id); 
$res_log->execute(); 
/删除 tb_shopping 表 
$query_shopping = 'delete from tb_shopping where id = :id'; 
$res_shopping = $pdo->prepare($query_shopping); 
S$res_shopping->bindParam(":id',$id); 
Sres_shopping->execute(); 
if($res_shopping->rowCount()X{ 
msg(1,' 操 作成 功 ','order.php'); 
jelsef 
msg(2,' 操 作 失 败 ','order.php'); 
} 





人 代码 贴 十 
@ bindParam: 该 函数 用 于 绑 定 参 数 ，:fahuo id 的 值 将 被 $fahuo id 替换 。 


上 述 代码 中 ， 首 先 接收 tb_shopping 表 的 ID， 然 后 查找 该 ID 所 在 记录 的 发 货 ID 字段 ， 接 着 根据 
发 货 ID 删除 tb_car log 车 辆 日 志 表 。 最 后 再 删除 tb_shopping 表 的 记录 。 


5.7.6 
国 


查看 发 货 单 明细 的 实现 过 程 


查看 发 货 单 明细 使 用 的 数据 表 : tb_car log、tb_shopping 


在 发 货 单 管理 页 面 中 ， 每 个 表格 内 只 显示 了 这 条 订单 的 主要 信息 ， 如 要 查看 订单 的 完整 信息 ， 可 
以 单 击 “ 查 看 明细 ”按钮 。 此 时 会 弹出 一 个 弹 窗 ， 弹 窗 包含 了 该 发 货 单 的 完整 信息 。 可 以 使 用 Layer 
插件 的 open0 方 法 实现 该 功能 ， 关 键 代码 如 下 : 


倒 程 11 代码 位 置 ， 资源 包 \TM\05\Logistics\order.php 


// 查 看 发 货 单 阴 细 
$('.show-order').click(function(X{ 


var order_id = $(this).attr('order_id'); // 获 取 发 货 单 ID 
layer.open({ // 调 用 open() 方 法 
/设置 参数 
type: 2, /弹出 层 类 型 
tile: ' 查 看 明细 '， // 弹 窗 标题 
shadeClose: true, /使 用 遮 罩 层 
shade: 0.5, // 遮 醒 比 例 
area: [960px', '90%'], // 弹 窗 大 小 


content: 'order_detail.php?id='+order_id 。 // 弹 窗 内 容 的 url 
入 





by 
用 
外 


第 5 章 物流 配送 信息 网 (Apache+PHP+MySQL 5.5 实现 ) 


Ah 代码 由 二 
@ content: content 参数 是 弹 窗 的 内 容 ， 内 容 来 源 于 order detailphp 页 面 ， 并 且 接 受 一 个 参数 。 


运行 效果 如 图 5.28 所 示 。 
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图 5.28 查看 发 货 单 明细 效果 
5.7.7 ”发 货 单打 印 的 实现 过 程 


国 ”发 货 单打 印 使 用 的 数据 表 : tb_car log、tb_shopping 

发 货 单 的 打印 是 为 货物 的 发 送 提供 一 个 书面 的 依据 ， 作 为 企业 与 客户 商业 活动 的 凭证 ， 其 内 容 必 
须 真实 、 准 确 。 单 击发 货 单 明 细 下 方 的 “打印 订单 ”按钮 ， 应 用 onClick 事件 调用 printpageO 自 定义 方 
法 ， 该 方法 调用 window.print0 方 法 实现 打印 功能 。 程 序 代码 如 下 : 


倒 程 12 ”代码 位 置 ， 资源 包 \TM\05\Logistics\order_detail.php 


<div class="text-center" onclick="printpage()"> 
<button class="btn btn-primary"> 打 印 订单 </button> 


</div> 
<script type="text/javascript" onclick="printpage()"> 
function printpage() 
{ 
window.print() // 调 用 打印 功能 
</script> 


运行 结果 如 图 5.29 所 示 。 
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图 5.29 打印 预览 效果 


5.7.8 回执 单 验收 的 实现 过 程 


国 回执 单 验收 使 用 的 数据 表 : tb_car log、tb_shopping 
回执 单 验收 的 主要 功能 就 是 对 货物 配送 完成 
确认 : 将 该 发 货 单 执行 类 型 更 新 为 “1”， 表 明 该 织 回执 音 
次 物流 配送 已 经 完成 ， 清 空 该 车 辆 的 使 用 日 志 ， 
便于 执行 下 一 个 订单 。 该 模块 的 业务 操作 流程 如 





图 5.30 所 示 。 

在 回执 单 验 收 的 实现 过 程 中 ， 首 先 需要 判断 T 
该 回执 单 状态 ， 如 果 回 执 单 已 经 确认 ， 在 发 货 单 

管理 模块 中 ， 回 执 状态 为 “已 确认 ”“ 回 执 确认 ” 
按钮 的 颜色 为 灰色 ， 且 按钮 不 可 单 击 。 和 否则 ， 回 图 5.30 回执 单 验收 的 业务 操作 流程 
执 状 态 为 “未 确认 ”“ 回 执 确认 ”按钮 的 颜色 为 

黄色 ， 按 钮 可 单 击 ， 页 面 效 果 如 图 5.31 所 示 。 


管理 员 








图 5.31 回执 单 状态 效果 
单 击 回执 状态 为 “未 确认 ”的 按钮 ， 弹 出 确认 提示 框 ， 单 击 “ 确 定 ” 按 钮 ， 则 使 用 Ajax 方式 异步 


全 
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提交 数据 ， 将 tb_shopping 表 的 ys 字段 值 更 改 为 1， 并 且 删 除 对 应 的 车 辆 使 用 信息 。 关 键 代码 如 下 : 
倒 程 13 代码 位 置 资源 包 \TM\05\Logistics\order.php 


// 确 认 发 货 单 
$('.confirm-order').on('click',function() { 
var order_id = $(this).attr('‘order_id'); // 获 取 发 货 单 ID 
varfahuo_id = $lthis).attr(fahuo_id'); /获取 发 货 ID 
layerconfirm(' 确 认 发 货 单 回执 ?,function(){f // 调 用 Layer 插件 的 confirm() 方 法 
/异步 提交 数据 
$.post('confirm_orderphp', {id:order_id,fahuo_id:fahuo_id}, function(res) { 
if(resX{ /判断 返回 值 
layer.msg(' 更 改 成 功 '); // 输 出 提示 信息 
window.location.href = "order.php"; // 页 面 跳 转 
}else{ 
layer.msg(' 更 改 失 败 '); // 输 出 提示 信息 
上 
六 
入 
return false; 
D; 





上 述 代码 中 ， 使 用 $.post0 方 法 将 order id 和 fahuo_id 两 个 值 传递 到 confirm_order.php 文件 , 在 
confirm_order.php 文件 中 执行 业务 逻辑 ， 关 键 代码 如 下 : 


倒 程 14 代码 位 置 ， 资源 包 \TM\05\Logistics\confirm order.php 





<?php 
include('Conn\config.php’); 
include('Library\function.php'); 
Sid = intval($_POST[id]); 
S$fahuo_id = $_POST[fahuo_id']; 
if(empty($id)X{ 
msg(2, 非 法 操作 ,orderphp'); 


> 
/清空 车 辆 日 志 
$query_delete ="delete from tb_car_log where fahuo_id='$fahuo_id"; 
$res_delete = $pdo->prepare($query_delete); 
Sres_delete->execute(); 
// 更 改 状态 
$query = 'update tb_shopping set fahuo_ys = 1 where id = ".$id; 
Sres = $pdo->prepare($query); 
Sres->execute(); 
if($res->rowCount()X 
echo True; 
Jelse{ 
echo False; 
} 


?> 
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上 述 代码 中 , 首先 根据 发 货 ID 删除 车 辆 的 使 用 日 志 , 然后 根据 发 货 单 ID 更 改 tb_shopping 表 状 态 。 
页 面 运行 结果 如 图 5.32 和 图 5.33 所 示 。 








图 5.32 回执 单 确认 
直入 必 和 闪 三 妥 全 和 人 
发 次 单 号 奈 罗 号 和 车主 电话 发 上 人 ”发现 人 号 话 
20171130114532 高 A69KE1 1979241m E23 1691044™"™ 
201790105104 AMIN 601044m E04 
201712909375621 A8747 。 1asTagere tel 
201712617141229 地 A59KE1 891044™™™ 张 三 1691044~~ 





图 5.33 ”回执 确认 结果 





5.8 基础 信息 管理 模块 设计 


Wn 


皇 
| 
E 
| 视频 讲解 


5.8.1 基础 信息 管理 模块 概述 











基础 信息 管理 模块 主要 包括 客户 信息 管理 , 用 于 
添加 客户 信息 和 删除 客户 信息 车 源 信息 管理 ， 主 要 世人 
用 于 对 车 源 信息 进行 管理 ， 实 现 车 源 信息 的 添加 、 修 
改 和 删除 功能 ， 管理 员 信息 管理 ,用 于 修改 管理 员 的 
密码 。 基 础 信息 管理 模块 的 功能 结果 如 图 5.34 所 示 。 有 
Ea 
5.8.2 ”管理 员 信 息 管理 模块 的 实现 过 程 
管理 员 

国 管理 员 信息 管理 模块 使 用 的 数据 表 : tb admin 

修改 管理 员 密码 的 模块 中 ， 需 要 答 入 管理 员 名 、 
原始 密码 、 新 密码 以 及 确认 密码 。 首 先 需要 在 前 端 检 
测 原始 密码 和 新 密码 不 能 相同 , 其 次 要 检测 原始 密码 人 


@ 
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和 确认 密码 是 否 一 致 。 关 键 代码 如 下 : 


倒 程 15 代码 位 置 资源 包 \TM\05\Logistics\change_pwd.php 
$(function() { 
S$('#password-form').submit(function() { 
if(!IcheckField( 类 admin_user',' 请 输入 管理 员 名 !")){ 
return false; 


} 
if(IcheckField(#admin_pass',' 请 输入 原始 密码 ')){ 
return false; 


} 
ilcheckField(#admin_new_pass', 请 输入 原始 密码 )){ 
return false; 


} 

1/ 判断 新 密码 与 原始 密码 是 否 相同 

if($(#admin_pass').val() == $(#admin_new_pass').val()){ 
layer.tips(' 新 密码 与 原始 密码 不 应 相同 ', #admin_new_pass', {time: 2000, tips: 2}); 
S$("#admin_new_pass").focus(); 
return false; 


} 

// 判 断 两 次 输入 新 密码 是 否 一 致 

if($(#admin_new_pass').val() != $('#admin_new_pass2').val()) { 
layer.tips(' 两 次 输入 密码 不 一 致 , #admin_new_pass2' {time: 2000, tips: 2}); 
$("#admin_new_pass2").focus(); 
return false; 

} 

return true; 

) 
D); 


上 述 代 码 中 ， 使 用 Layer 插件 的 tips0 方 法 弹出 提示 信息 ， 运 行 效 果 如 图 5.35 所 示 。 


修改 三 





第 理 呈 








图 5.35 检测 输入 信息 
更 改 密码 的 功能 主要 是 通过 store pwd.php 文件 实现 的 ， 在 这 个 文件 中 ， 首 先 判断 用 户 输入 的 管理 
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员 和 原始 密码 是 否 匹 配 ， 如 果 不 匹 配 ， 提 示 “ 原 始 密码 错误 !”， 如 果 匹 配 ， 则 更 改 原始 密码 为 新 密码 。 
关键 代码 如 下 : 


倒 程 16 代码 位 置 资源 包 \TM\05\Logistics\store pwd.php 


<?php 
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include('Conn\config.php'); 
include('Libraryfunction.php '); 
// 判 断 是 否 登录 
if(IcheckLogin()X{ 
msg(2," 请 先 登录 ',login.php'); 
} /接收 数据 */ 
S$admin_user = $_POST[admin_user]; 
$admin_pass = md5($_POST[admin_pass]); 
$admin_new_pass =md5($_POST[admin_new_pass]); 


$admin_new_pass2 = md5($_POST[admin_new_pass2"); 


/* 后 台 验 证 数据 */ 
if(empty($admin_user)) 


msg(2, ' 请 输入 管理 员 名 ! '); 
} 
ifempty($admin_pass)) 


msg(2, ' 请 输入 原始 密码 !"); 


} 
if(empty($admin_new_pass)) 


msg(2, ' 请 输入 新 密码 !"); 
} 
if($admin_new_pass == $admin_pass}{ 
msg(2, 新 密码 与 原始 密码 不 应 相同 '); 
站 


if($admin_new_pass !== $admin_new_pass2){ 
msg(2, ' 两 次 输入 密码 不 一 致 "); 


} 
/* 判 断 原始 密码 是 否 正确 **/ 


/引入 配置 文件 
/引入 函数 库 


/使 用 md5 加 密 
/使 用 md5 加 密 
/使 用 md5 加 密 


$query = "select * from tb_admin where admin_user = '$admin_user and admin_pass = '$admin_pass"; 


Sres = $pdo->prepare($query); 
Sres->execute(); 
$count = $res->fetchColumn(); 
if(empty($count)X{ 

msg(2, "原始 密码 错误 !",'change_pwd.php'); 


} 
/更 改 tb_admin 表 


$query = "update tb_admin set admin_pass = '$admin_new_pass' where admin_user ='$admin_user "; 


S$update = $pdo->prepare($query); 
Supdate->execute(); 
if($update->rowCount()X{ 
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msg(1,' 操 作成 功 ','change_pwd.php'); 
}else{ 
msg(2,' 操 作 失 败 ','change_pwd.php'); 


?> 





上 述 代码 中 ， 使 用 md5 对 输入 的 密码 加 密 ， 再 将 加 密 后 的 结果 与 数据 库 中 数据 进行 对 比 。 


Ah 代码 贴 二 
@ fetchColumn: 从 结果 集中 的 下 一 行 返回 单独 的 一 列 。 
@ update: 用 于 更 新 数据 表 中 的 字段 值 . 
上 自 rowCount: 返回 受 影响 的 行 数 。 


5.8.3 客户 信息 管理 的 实现 过 程 


国 。 客户 信息 管理 模块 使 用 的 数据 表 : tb_customer 
客户 信息 管理 模块 的 主要 功能 就 是 向 数据 库 中 添加 客户 信息 ， 并 且 可 以 对 指定 的 客户 进行 删除 操 
作 。 客 户 信息 管理 模块 的 运行 结果 如 图 5.36 所 示 。 


+ 加 
mm i EIE3 


5.36 客户 信息 管理 模块 的 运行 结果 


客户 信息 管理 模块 主要 通过 customerphp、add_customerphp 和 delete_customerphp 文件 来 完成 。 
首先 通过 customerphp 文件 来 输出 数据 库 中 的 所 有 客户 信息 ， 设 置 查找 、 新 增 、 编 辑 和 删除 客户 信息 
的 超 链接 。 其 输出 客户 详细 信息 的 关键 代码 如 下 : 


倒 程 17 ”代码 位 置 ， 资源 包 \TM\05\Logistics\customer.php 
<tbody> 
<?php while($row = $res->fetch(PDO::FETCH_ASSOC)){ ?> 
<tr> 
<td><?php echo $row['customer_user] ?></td> 
<td><?php echo $row['customer_tel] ?></td> 
<td><?php echo $row['customer_address] ?></td> 
<td class="col-md-2"> 
<a href="add_customer.php?id=<?php echo $row['customer_id]?>"> 
<button class="btn btn-info " type="button"><i class="glyphicon glyphicon-edit"></i> 
&nbsp; 编 辑 </button> 
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</a> 
<a href="delete_customer.php?id=<?php echo $row['customer_id]?>" class="del"> 
<button class="btn btn-danger " type="button"><i class="glyphicon glyphicon-trash"></i> 
&nbsp; 删 除 </button> 
</a> 
</td> 
</tr> 
<?php } ?> 
</tbody> 





客户 信息 添加 和 编辑 的 操作 通过 add_customerphp 文件 完成 , 首先 连接 数据 库 , 然后 判断 提交 的 电 
话 号 码 格式 是 否 正确 ， 接 着 判断 是 新 增 操作 或 是 编辑 操作 ， 若 人 D 存在 是 编辑 操作 ， 根 据 客户 ID 获取 
客户 信息 ， 否 则 为 新 增 操作 。 最 后 将 客户 信息 数据 提交 到 数据 库 中 。 程 序 的 关键 代码 如 下 : 


倒 程 18 代码 位 置 资源 包 \TM\05\Logistics\add_customer.php 


<?php 

include('Conn\config.php'); /1/ 引 入 配置 文件 
include('Library\function.php’); /1/ 引 入 函数 库 
/| 判断 是 否 登录 

if(IcheckLogin()X{ 


msg(2," 请 先 登 录 ','login.php'); 
} 
$customer_id = isset($_GET[id]) ?$_GET['id] : 0; 
$query = 'select * from tb_customer where customer_id = .$customer_id; 
Sres = $pdo->prepare($query); 
Sres->execute(); 
S$row = $res->fetch(PDO::FETCH_ASSOC); 
?> 


页 面 运行 效 果 如 图 5.37 所 示 。 


物流 配送 信息 网 





图 5.37 编辑 页 面 的 运行 结果 


客户 信息 删除 的 操作 通过 delete_customer.php 文件 来 完成 , 主要 以 超 链 接 中 提交 的 变量 为 根据 , 删 
除数 据 库 中 指定 的 客户 信息 。 程 序 的 关键 代码 如 下 : 


局 
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倒 程 19 代码 位 置 资源 包 \TM\05\Logistics\delete_customer.php 
<?php 
include('Conn\config.php’); // 引 入 配置 文件 
include('Library\function.php'); 1 引入 函数 库 
// 判 断 是 否 登录 
if(IcheckLogin()){ 
msg(2," 请 先 登录 ',login.php'); 





} 
$customer_id =$_GET[Tid]; /获取 客户 id 
iflempty($customer_id))f 

msg(2,' 非 法 操作 ','customer.php'); 


} 
/™ 执 行 删 除 操作 **/ 
$query = 'delete from tb_customer where customer_id = :id'; 
Sres = $pdo->prepare($query); 
S$res->bindParam(":id', $customer_id); 
Sres->execute(); 
if($res->rowCount()X{ 

msg(1,' 删 除 成 功 ','customer.php'); 
Jelse{ 

msg(2,' 删 除 失败 '); 
} 


reg 





5.8.4 ”车 源 信息 管理 的 实现 过 程 


国 ” 车 源 信息 管理 模块 使 用 的 数据 表 : tb_car 
车 源 信息 管理 模块 主要 实现 对 车 辆 信息 的 添加 、 修 改 和 删除 操作 。 车 源 信息 管理 模块 的 运行 结果 
如 图 5.38 所 示 。 


rr 


车主 姓名 车 主 刁 份 证 号 坪 昌 二 车 主 电 话 。 地址 。 车 和 中 法 诗 允 扩 操作 
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图 5.38 车 源 信息 管理 模块 的 运行 结果 
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车 源 信息 管理 模块 主要 由 3 个 文件 组 成 。carphp 文件 用 于 输出 车 辆 的 信息 和 搜索 车 辆 的 信息 。 根 
据 车 牌号 码 从 数据 库 中 读 取 对 应 车 辆 的 信息 , 进而 实现 对 该 车 辆 信息 的 修改 或 者 删除 操作 ; add_carphp 
文件 用 于 添加 和 编辑 车 辆 信息 到 数据 库 。delete php 文件 用 于 删除 指定 的 车 辆 信息 。add_carphp 文件 中 


Form 表单 使 用 的 重要 元 素 如 表 5.9 所 示 。 
表 5.9 车 源 信息 管理 模块 中 使 用 的 重要 表单 元 素 


























名 称 元 素 类 型 重要 属性 含义 
add-form form action="store_car.php" method="post" 表单 
Username text name="usemame" value="<?php echo $row['username’] ?>" 车 主 姓名 
car_number text name="car number" value="<?php echo Srow['car number]?>" 车 牌号 码 
car tel text name="tel" value="<?php echo Srow['el] ?>" 车 主 电 话 
user number text name="user_number" value="<?php echo $row['user_ number]?>" 身份 证 号 码 
car_content textarea <?php echo $row['car content] ?> 车 辆 描述 
address textarea name="address" value="<?php echo $row['address'] ?>" 家 庭 地 址 
car road textarea <?php echo $row['car road'] ?> 运输 路 线 


添加 和 编辑 车 源 信息 的 功能 如 图 5.39 所 示 。 





mts 王牌 所 码 吉 A89KE1 身份 证 二 22072419670222" 
电 1379211 主 地 址 长 看 市 
长 春 -北京 重 计 


伟大 德 电 - 杰 驰 疡 Actros 2642 LS DNA 5X2 公路 雄 引 夺 


图 5.39 编辑 车 源 信息 的 运行 结果 


5.9 ”开发 技巧 与 难点 分 析 


5.9.1 应 用 Session 存储 有 关 用 户 会 话 的 信息 


在 物流 配送 信息 网 系统 中 , 判断 管理 员 是 否 登 录 是 通过 Session 会 话 来 实现 的 。 下 面 就 来 讲解 一 下 


Session 会 话 的 相关 知识 grid Session。 


当 运 行 一 个 应 用 程序 时 ， 经 常会 打开 它 ， 做 些 更 改 ， 然 后 关闭 它 。 这 很 像 一 次 会 话 。 计 算 机 清楚 
你 是 谁 。 它 知 道 你 何 时 让 动 应 用 程序 并 在 何 时 终止 。 但 是 在 因特网 上 ， 存 在 一 个 问题 : 服务 器 不 知 
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道 你 是 谁 以 及 你 做 什么 ， 这 是 由 于 HTTP 地 址 不 能 维持 状态 。 

Session 通过 在 服务 器 上 存储 用 户 信息 解决 了 这 个 问题 。 不 过 ， 会 话 信息 是 临时 的 ， 在 用 户 离开 网 
站 后 将 被 删除 。Session 的 工作 机 制 是 : 为 每 个 访问 者 创建 一 个 唯一 的 id (UID)， 并 基于 这 个 UID 来 
存储 变量 。UID 存储 在 Cookie 中 ， 亦 或 通过 URL 进行 传导 。 

Session 的 存储 和 读 取 非常 简单 ， 首 先 使 用 session_start0 函 数 来 开启 Session， 如 果 将 $user 变量 存 
储 到 Session， 可 以 按 如 下 方式 存储 Session 变量 : 


$_Session[user] = $user 


在 登录 页 面 ， 当 用 户 输入 正确 的 用 户 名 和 密码 后 ， 需 要 将 用 户 信息 存储 到 Session， 以 方便 后 面 判 
断 用 户 登 录 状 态 。 程 序 的 关键 代码 如 下 : 


倒 程 20 代码 位 置 ， 资源 包 \TM\05\Logistics\login.php 
/判断 账户 密码 是 否 正确 */ 
$query = "select * from tb_admin where admin_user = :user and admin_pass = :password"; 
Sres = $pdo->prepare($query); 
S$res->execute(array(":user'’=>$admin_user,":password'=>md5($admin_pass))); 
$admin = $res->fetch(PDO::FETCH_ASSOC); 
ifis_array($admin) && lempty($admin))f 
@ session_start(); 
@ $_SESSION[user] = $admin; 
echo true; 
jelsef 
echo false; 


} 





< 代码 贴 二 
@ session_ start: 该 函数 用 于 开启 Session。 
@ $_SESSION['user]: 存储 $admin 变量 ， 该 变量 是 一 个 数组 。 


接 下 来 读 取 Session， 可 以 直接 使 用 $_SESSION[mser] 来 获取 。 如 果 $_SESSION[mser] 存 在 ， 则 表 
示 用 户 已 经 登录 ， 否 则 未 登录 。 由 于 每 一 个 需要 登录 权限 的 页 面 都 需 判断 $_SESSION["user"] 是 否 存在 ， 
所 以 定义 一 个 函数 来 简化 代码 ， 程 序 代码 如 下 : 


倒 程 21 代码 位 置 ， 资源 包 \TM\05\Logistics\Library\function.php 


function checkLogin() 
‘ 

/开启 session 

session_start(); 

/用户 未 登录 

if(lisset($_SESSION['user') ||empty($_SESSIONTuser])) 

{ 

return false; 


} 
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return true; 
} 


在 需要 登录 权限 的 页 面 ， 直 接 调 用 checkLogin0 函 数 即 可 。 例如， 首页 判断 管理 员 是 否 登 录 ， 如 未 
登录 ， 跳 转 到 登录 页 ， 关 键 代 码 如 下 : 


倒 程 22 代码 位 置 ， 资源 包 \TM\05\Logistics\index.php 


<?php 
include('Library\function.php’); 
// 判 断 是 否 登录 
if(checkLogin()) 


msg(1,' 您 已 登录 ','index.php'); 


5.9.2 ”应 用 正则 表达 式 验证 电话 号 码 


在 物流 配送 信息 网 中 ， 确 认输 入 的 电话 号 码 正 确 与 否 是 一 个 非常 重要 的 问题 ， 如 果 客 户 在 填写 订 
单 时 使 用 了 错误 的 电话 号 码 ， 那 么 就 会 带 来 很 多 不 必要 的 麻烦 ， 所 以 在 填写 电话 号 码 时 一 定 要 注意 电 
话 号 码 的 准确 性 。 为 了 更 好 地 确保 电话 号 码 输入 的 准确 性 ， 可 以 通过 程序 对 其 进行 控制 。 虽 然 程序 不 
能 判断 电话 号 码 输入 得 是 否 正确 ， 但 是 通过 程序 可 以 对 电话 号 码 的 格式 进行 判断 ， 从 而 避免 一 些 电话 
号 码 在 录入 的 过 程 中 出 现 多 写 或 者 漏 写 数字 的 错误 。 

验证 电话 号 码 格式 是 否 正确 可 以 通过 正则 表达 式 来 完成 。 通 过 正则 表达 式 可 以 有 效 地 判断 输入 电 
话 号 码 的 格式 是 否 正确 。 

在 进行 判断 之 前 首先 要 对 电话 号 码 的 格式 进行 分 类 : 第 一 类 区 号 是 3 位 数字 ， 其 他 是 8 位 数字 ; 
第 2 类 区 号 是 4 位 数字 ， 其 他 是 7 位 数字 ， 第 3 类 区 号 是 4 位 数字 ， 其 他 是 8 位 数字 ， 第 4 类 是 手机 
电话 号 码 ， 由 11 位 数字 组 成 。 

然后 编写 一 个 判断 电话 号 码 格式 的 正则 表达 式 ， 其 代码 如 下 : 


MQd{3})(d{8))SI Nd{4}) 7S dA4}) Nd{8})s Nd{11))s/ 


正则 表达 式 的 功能 分 析 如 下 : 使 用 “^” 和 “$” 对 字符 串 进行 边界 限制 ， 对 区 号 从 字符 串 的 开始 
进行 匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 ; 将 括号 “QO” 中 的 内 容 作 为 一 个 原子 使 用 ; 使 用 
“\d” 来 匹配 一 个 数字 ， 区 号 为 3 或 4 个 数字 ， 其 他 数字 为 7 或 8 个 ， 或 者 直接 判断 为 11 个 数字 ; 使 
用 “{}” 来 对 前 字符 进行 重复 匹配 ， 使 用 “|” 对 匹配 的 模式 进行 选择 ， 分 成 4 个 模式 。 

正则 表达 式 创建 完成 后 就 可 以 应 用 到 程序 中 对 电话 号 码 进行 判断 ， 要 完成 对 电话 号 码 的 判断 还 需 
要 应 用 preg_matchO 函 数 。 

preg_match(O) 函 数 对 正则 表达 式 进 行 匹 配 。 返 回 匹 配 的 次 数 ，0 次 (没有 匹配 ) 或 者 1 次 ， 如 果 出 
错 则 返回 false。preg_match0O 函 数 的 参数 说 明 如 表 5.10 所 示 。 


@ 
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表 5.10 ”preg_match() 函 数 的 参数 说 明 

参 。 数 说 明 

pattem | 必要 参数 ， 指 定 匹配 的 正则 表达 式 

subject 。 | 必要 参数 ， 指 定 要 搜索 的 字符 串 
可 选 参数 ， 如 果 提 供 了 subpattems， 则 其 会 被 搜索 的 结果 所 填充 。$ subpattems[0] 将 包含 与 整个 模式 匹 
配 的 文本 ，$ subpatterns[1] 将 包含 与 第 一 个 捕获 的 括号 中 的 子 模式 所 匹配 的 文本 ， 以 此 类 扒 
可 选 参数 , 如 果 设 定 标记 PREG_OFFSET_CAPTURE， 则 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 
串 偏 移 量 。 注 意 这 改变 了 返回 数组 的 值 ， 使 其 中 的 每 个 单元 也 是 一 个 数组 ， 其 中 第 一 项 为 匹配 字符 串 ， 
第 二 项 为 其 偏 移 量 。 本 参数 自 PHP 4.3.0 开始 使 用 


最 后 将 正则 表达 式 和 preg_match0 函 数 应 用 到 实际 的 程序 中 , 实现 对 电话 号 码 格式 的 判断 。 这 里 以 
store_order.php 文件 中 的 代码 为 例 ， 关 键 代码 如 下 : 


倒 程 23 代码 位 置 ， 资源 包 \TM\05\Logistics\store_order.php 
<?php 


S$rule = A(d(3F)Gdf8D)SIACd{4}-)Cd(7DSIA(d(4F-)Cdf8)SIACd(115$7 
if(!preg_match($rule,$car_tel)) 











subpatterns 













flags 


msg(2,' 车 主 电话 格式 错误 ! '); 





5.10 本 章 总 结 


本 章 从 某 物流 配送 公司 的 实际 需求 角度 出 发 ， 开 发 一 个 完整 的 物流 配送 信息 管理 系统 ， 详 细 地 讲 
解 了 物流 配送 信息 网 的 开发 流程 ， 从 最 初 的 需求 分 析 、 可 行 性 分 析 ， 到 系统 的 设计 、 数 据 库 的 设计 ， 
其 中 重点 突出 车 源 信息 查询 模块 、 发 货 单 管理 模块 和 回执 单 验收 管理 模块 的 设计 。 通 过 本 章 的 学 习 ， 
读者 不 但 可 以 了 解 物流 配送 系统 开发 的 整体 思路 ， 而 且 能 够 掌握 很 多 关键 的 技术 和 技巧 : 函数 的 灵活 
运用 、MySQL 存储 过 程 的 创建 和 应 用 、 报 表 打印 技术 的 实现 和 应 用 正则 表达 式 验证 电话 号 码 等 。 


全 后 二 
后 





学 校 图 书馆 管理 系统 


(Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


随 着 网 络 技术 的 高 速 发 展 和 计算 机 应 用 的 善 及 ， 利 用 计算 机 对 图 
书馆 的 日 常 工作 进行 管理 势 在 必 行 。 虽 然 目前 很 多 大 型 的 图 书馆 已 经 
有 一 整套 比较 完善 的 管理 系统 ， 但 是 在 一 些 中 小 型 的 图 书馆 中 ， 大 部 
分 工作 仍 需 由 手工 完成 ， 工 作 起 来 效率 比较 低 ， 管 理 员 不 能 及 时 了 解 
图 书馆 内 各 类 图 书 的 借阅 情况 ,读者 需要 的 图 书 难以 在 和 经 时 间 内 找到 ， 
不 便于 动态 及 时 地 调整 图 书 结构 。 为 了 更 好 地 适应 当前 读者 的 借阅 需 
求 ， 解 决 手工 管理 中 存在 的 许多 琴 端 ， 越 来 越 多 的 中 小 型 图 书馆 正在 
逐步 向 计算 机 信息 化 管理 转变 。 本 章 通 过 开发 一 个 流行 的 图 书馆 管理 
系统 ， 为 读者 讲解 详细 的 项 目 开发 流程 。 

通过 阅读 本 章 ， 可 以 学 习 到 

mW 图 书馆 管理 系统 天 发 的 基本 过 程 

MW 系统 设计 的 方法 

让 如 何 分 析 并 设计 数据 库 、 数 据 表 

让 多 表 查 询 的 方法 

MW 面向 对 象 的 编程 方法 

Wm 主要 功能 模块 的 实现 方法 

由 单元 测试 的 方法 

Hi 如 何 自 动 计算 图 书 娄 还 日 期 

由 ”内 联接 和 外 联接 语句 的 使 用 方法 
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6.1 开发 背景 





视频 讲解 

知 源 图 书馆 是 一 家 吉林 X X 大 学 的 大 型 图 书馆 。 随 着 学 校 图 书馆 规模 的 不 断 壮 大 ， ss 
经 营 的 图 书 品种 、 数 量 也 逐渐 增多 。 在 学 校 图 书馆 不 断 发 展 的 同时 ， 校 图 书馆 常年 采用 的 传统 的 人 工 
方式 管理 暴露 了 一 些 问 题 。 例 如 ， 查 找 读者 借阅 的 某 一 本 图 书 的 具体 摆 放 位 置 ， 需 要 靠 人 工 记忆 在 书 
海中 苦 苦 查找 ， 由 于 图 书 储存 量 大 ， 很 难 准确 定位 图 书 的 具体 位 置 ， 因 此 每 天 都 要 浪费 大 量 宝贵 的 时 
间 资 源 。 学 校 图 书馆 为 提高 工作 效率 ， 同 时 摆脱 图 书 管理 人 员 在 工作 中 出 现 的 种 种 弊端 ， 现 需要 委托 
某 单位 开发 一 个 学 校 图 书馆 管理 系统 。 


6.2 需求 分 析 


通过 计算 机 对 图 书 进行 管理 ， 不 仅 为 图 书馆 的 管理 注入 了 新 的 生机 ， 而 且 在 运营 过 程 中 还 节省 了 
大 量 的 人 力 、 物 力 、 财 力 和 时 间 ; 不 仅 提高 了 图 书馆 的 工作 效率 ， 还 为 图 书馆 在 读者 群 中 树立 了 一 个 
全 新 的 形象 ， 为 图 书馆 日 后 发 展 莫 定 了 一 个 良好 的 基础 。 通 过 对 一 些 大 型 图 书馆 的 实际 考察 、 分 析 ， 
并 结合 图 书馆 的 要 求 以 及 实际 的 市 场 调查 ， 要 求 本 系统 具有 以 下 功能 : 
网 站 设计 页 面 要 求 美观 大 方 、 个 性 化 ， 功 能 全 面 ， 操 作 简单 。 
要 求实 现 基础 信息 的 管理 平台 。 
要 求 对 所 有 读者 进行 管理 。 
要 求实 现 图 书 借阅 排行 、 了 解 当前 的 畅销 书 。 
商品 分 类 详尽 ， 可 按 不 同类 别 查看 图 书信 息 。 
提供 快速 的 图 书信 息 、 图 书 借阅 检索 功能 ， 保 证 数据 查询 的 灵活 性 。 
实现 图 书 借阅 、 图 书 续 借 、 图 书 归还 的 功能 。 
实现 综合 条 件 查询 ， 如 按 用 户 指定 条 件 查询 、 按 日 期 时 间 段 查询 、 综 合 条 件 查询 等 。 
要 求 图 书 借阅 、 续 借 、 归 还 时 记 下 每 一 笔记 录 的 操作 员 。 
实现 对 图 书 借阅 、 续 借 和 归还 过 程 的 全 程 数 据 信息 跟踪 。 
提供 借阅 到 期 提醒 功能 ， 使 管理 者 可 以 及 时 了 解 到 已 经 到 达 归 还 日 期 的 图 书 借阅 信息 。 
提供 灵活 、 方 便 的 权限 设置 功能 ， 使 整个 系统 的 管理 分 工 明确 。 
有 具有 易 维护 性 和 易 操作 性 。 


办 办 办 办 办 办 办 办 办 凶 凶 凶 


6.3 系统 设计 


6.3.1 系统 目标 


根据 前 面 所作 的 需求 分 析 及 用 户 的 需求 可 以 得 出 ， 学 校 图 书馆 管理 系统 实施 后 ， 应 达到 以 下 


_ 国 
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目标 : 

网 站 设计 页 面 要 求 美观 大 方 、 功 能 全 面 ， 操 作 简单 。 

网 站 整体 结构 和 操作 流程 合理 顺畅 ， 实 现 人 性 化 设计 。 
规范 、 完 善 的 基础 信息 设置 。 

对 操作 员 设 置 不 同 的 操作 权限 ， 为 管理 员 提供 修改 权限 功能 。 
对 所 有 读者 进行 集中 管理 。 

对 图 书信 息 进行 集中 管理 。 

实现 图 书 借阅 排行 ， 以 便 了 解 当前 的 畅销 书 。 

提供 快速 的 图 书信 息 、 图 书 借阅 检索 功能 。 

实现 图 书 借阅 、 图 书 续 借 、 图 书 归还 功能 。 

实现 综合 条 件 查询 ， 如 按 用 户 指定 条 件 查询 、 按 日 期 时 间 段 查询 、 综 合 条 件 查 询 等 。 
实现 图 书 借阅 、 续 借 、 归 还 时 记 下 每 一 笔记 录 的 操作 员 。 
支持 图 书 到 期 提醒 功能 。 

为 操作 员 提供 密码 修改 功能 。 

系统 运行 稳定 、 安 全 可 靠 。 


6.3.2 ”系统 功能 结构 


根据 学 校 图 书馆 管理 系统 的 特点 ， 可 以 将 其 分 为 系统 设置 、 读 者 管理 、 图 书 档案 管理 、 图 书 借 还 、 
系统 查询 等 5 个 部 分 ， 其 中 各 个 部 分 及 其 包括 的 具体 功能 模块 如 图 6.1 所 示 。 


学 校 图 书馆 管理 系统 


辐 同 回回 网 加 加 网 加 回回 加 回回 


证 二 该 击 困 
贿 束 泗 彤 于 
贿 焉 沙 
贿 束 洪 击 


鸿 哄 洋 并 红 苹 
鸿 叶 洒 庚 疆 骂 
西 卉 击 困 
栈 游 薄 力 
疝 和 远 巩 力 
格 界 洲 获 届 困 
熙 内 一 著 水 国 
山 溃 沙 业 西 审 
今世 呀 册 











图 6.1 学 校 图 书馆 管理 系统 功能 结构 图 
6.3.3 ”系统 流程 图 
学 校 图 书馆 管理 系统 的 流程 如 图 6.2 所 示 。 


他- 
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图 书 借阅 图 书 续 借 图 书 归还 
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y 
图 书 归还 档案 














图 62 学 校 图 书馆 管理 系统 流程 图 
6.3.4 系统 预览 


学 校 图 书馆 管理 系统 由 多 个 程序 页 面 组 成 ， 下 面 仅 列 出 几 个 典型 页 面 ， 其 他 页 面 参 见 资源 包 中 的 
源 程 序 。 

系统 登录 页 面 如 图 6.3 所 示 ， 该 页 面 用 于 实现 管理 员 登 录 。 系 统 首页 如 图 6.4 所 示 ， 该 页 面 用 于 实 
现 显示 系统 导航 、 图 书 借阅 排行 和 版 权 信息 等 功能 。 





1 sae 
区 加 办 sm 提 才 计价 元) 作风 禾 
3 gheeeTr? Winal bad 控 作 参考 大 二 。 计划 外 程序 设计 Wop 书 某 。 人 民 闻 电导 。 明科 执 。 ao。 了 












2 pe 宝 风 系 列 。。F 书 条 ”人民 部 志 4 朵 村 。 用 晶 科 盾 。 名 .0 
52 PE 机 人才 全 全 各 肝 学 技术。 PF 书 染 人 民 站 沁 术 守明 上 科技 。 交 0 
lss PE 从 三 计算 机 程 订 设 计 Pr 书 某 清 华 大 学 卫 必 守明 日 和 法 

5 em FBR¥rale 于 扫 达 粮 术 。。 其 他 。 训 朵 太 尝 上 计 “明科 技 St. 04 
as 计 本 机 程序 设计 Pr 书 条 二 大学 出 且 导 明日 科 扫 

人。 34aeTsy PE 天 和 攻 系 坟 条 撩 元 生地 生 。 副 所 技术。 PE 书 半 。 人民 地 电导 。 才 晶 科 禾 。 外. 
risk P 本 开业 有 宇和 到 FE 包 条 。 少 大 尖 六 守明 科技 。 册 








寺 知 机 程 议 计 Pr 书 各 吉村 大 学 : 术 计 明日 科技 











图 63 系统 登录 (资源 包 \TM\06\ibrary\login php) 图 6.4 系统 首页 (资源 包 \TM\06\ibrary\index.php) 


图 书 借阅 页 面 如 图 6.5 所 示 ， 该 页 面 用 于 实现 图 书 借阅 功能 。 图 书 借阅 查询 页 面 如 图 6.6 所 示 ， 该 
页 面 用 于 实现 按照 复合 条 件 查询 图 书 借阅 信息 。 
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(资源 包 \TM\06\library\bookBorrow.php) (资源 包 \TM\06\library\borrowQuery.php) 


6.3.5 ”开发 环境 


在 开发 学 校 图 书馆 管理 系统 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows 7/Linux (推荐 )。 

服务 器 : Apache 2.4.18。 

PHP 软件 : PHP 7.0.12。 

数据 库 : MySQL 5.5.47。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-3.5.8。 
开发 工具 : PhpStorm 2016.3。 

浏览 器 : Google Chrome。 

分 辨 率 : 最 佳 效果 为 1680x1050 像素 。 


客户 端 


浏览 器 : Google Chrome。 
分 辨 率 : 最 佳 效果 为 1680x1050 像素 。 


6.3.6 ”文件 夹 组 织 结构 


在 编写 代码 之 前 , 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创 


罗 罗 NN 办 办 办 办 办 办 罗 








建 出 来 《例如 ， 创 建 一 个 名 为 Images 的 文件 夹 ， 用 于 保存 | 一 noahwonoaneaxn 
网 站 中 所 使 用 的 图 片 ), 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 人 RE 
也 可 以 规范 网 站 的 整体 架构 。 笔者 在 开发 学 校 图 书馆 管理 系 [一 一 townwane wom ptzn 





统 时 , 设计 了 如 图 6.7 所 示 的 文件 夹 组 织 结构 图 。 在 开发 时 ， 图 6.7 文件 夹 组 织 结构 
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只 需要 将 所 创建 的 文件 保存 在 相应 的 文件 夹 中 即 可 。 


6.4 数据 库 设 计 





学 校 图 书馆 管理 系统 是 一 个 数据 库 开发 的 Web 网 站 。 下 面 对 学 校 图 书馆 使 用 的 数 
据 库 进行 分 析 和 介绍 。 


6.4.1 数据 库 分 析 


由 于 本 系统 是 为 中 小 型 的 图 书馆 开发 的 程序 ， 需 要 充分 考虑 到 成 本 问题 及 使 用 需求 〈 如 跨 平 台 ) 
等 问题 ， 而 MySQL 是 世界 上 最 为 流行 的 开放 源码 的 数据 库 ， 是 完全 网 络 化 的 跨 平台 的 关系 型 数据 库 
系统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 


6.4.2 数据库 概 念 设计 


根据 以 上 各 节 对 系统 所 做 的 需求 分 析 、 系 统 设 计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实 体 分 别 为 图 书 
档案 实体 、 读 者 档案 实体 、 借 阅 档案 实体 、 归 还 档案 实体 和 管理 员 实体 。 下 面 将 介绍 几 个 关键 实体 的 
E-R 图 。 


1. 图 书 档案 实体 


图 书 档案 实体 包括 编号 、 条 形 码 、 书 名 、 类 型 、 作 者 、 译 者 、 出 版 社 、 价 格 、 页 码 、 书 架 、 录 入 
时 间 和 操作 员 等 属性 。 图 书 档案 实体 的 E-R 图 如 图 6.8 所 示 。 


2. 读者 档案 实体 


读者 档案 实体 包括 编号 、 姓 名 、 性 别 、 条 形 码 、 职 业 、 出 生日 期 、 有 效 证 件 、 证 件 号 码 、 电 话 、 
电子 邮件 、 登 记 日 期 、 操 作 员 、 类 型 和 备注 等 属性 。 读 者 档案 实体 的 E-R 图 如 图 6.9 所 示 。 
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图 6.8 图 书 档案 实体 的 E-R 图 图 6.9 读者 档案 实体 的 E-R 图 
3. 借阅 档案 实体 
借阅 档案 实体 包括 编号 、 读 者 编号 、 图 书 编号 、 借 书 时 间 、 应 还 时 间 、 操 作 员 和 是 否 归还 等 属性 。 


@ 
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借阅 档案 实体 的 E-R 图 如 图 6.10 所 示 。 
4. 归还 档案 实体 


归还 档案 实体 包括 编号 、 读 者 编号 、 图 书 编号 、 归 还 时 间 和 操作 员 等 属性 。 归 还 档案 实体 的 E-R 
图 如 图 6.11 所 示 。 


还 时 归还 


图 6.10 借阅 档案 实体 的 E-R 图 图 6.11 ”归还 档案 实体 的 E-R 图 
6.4.3 创建 数据 库 及 数据 表 


结合 实际 情况 及 对 用 户 需 求 的 分 析 , 学 校 图 书馆 管理 系统 db_library 数据 库 主 要 包含 11 个 数据 表 ， 
如 表 6.1 所 示 。 



































表 6.1 db_library 数据 库 中 的 数据 表 

















表 说 明 
tb bookcase 图 书 书架 信息 表 
th_bookinfo 图 书信 息 表 
tb booktype 图 书 类 型 信息 表 
th_borrow 图 书 借阅 信息 表 
tb_library utf8_general ci 图 书馆 信息 表 
tb_manager utf8_general ci 管理 员 信 息 表 
tb_parameter utf8_general ci 参数 设置 信息 表 
tb_publishing utf8 general ci 出 版 社 信息 表 
tb_purview utf8 general ci 权限 信息 表 
th_reader utf8 general ci 读者 信息 表 





th_readertype utf8 general ci 读者 类 型 信息 表 
结合 数据 表 的 创建 方法 ， 读 者 可 以 自行 创建 以 下 数据 表 。 
1. tb_bookinfo (图 书信 息 表 ) 
图 书信 息 表 主 要 用 于 存储 图 书 的 基础 信息 。 该 数据 表 的 结构 如 表 6.2 所 示 。 


@ 
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表 6.2 图 书信 息 表 结构 



































名 字 类 型 整 理 空 说 了 明 
barcode varchar(30) | utf8 general ci 是 图 书 条 形 码 
bookname varchar(70) | utf8 general ci 是 图 书 名 称 
typeid int(10) 是 图 书 类 型 
author varchar(30) | utf8 general ci 是 图 书 作 者 
translator varchar(30) | utf8 general ci 是 图 书 译 者 
ISBN varchar(20) | utf8 general ci 是 图 书 ISBN 
price float(8.2) 是 图 书 定价 
page int(10) 是 图 书页 码 
bookcase int(10) 是 图 书 书架 
storage int(10) 是 图 书库 存 
inTime date 入 库 时 间 
operator varchar(30) | utfg_general ci 操作 员 
del tinyint(1) 是 否 删 除 
id int(11) 自动 编号 








2. tb_borrow (图 书 借阅 信息 表 ) 
图 书 借阅 信息 表 主 要 用 于 存储 图 书 的 借阅 信息 。 该 数据 表 的 结构 如 表 6.3 所 示 。 
表 6.3 图 书 借阅 信息 表 结 构 














名 字 | 类 型 | 整理 | 空 | 默认 | 额外 | 说 明 
id int(10) 自动 编号 
Teaderid int(10) 读者 编号 
bookid int(10) 图 书 编号 
borrowTime | date 图 书 借阅 时 间 
backTime date 图 书 归 还 时 间 
operator varchar(30) | utf8 general ci 操作 员 
ifback tinyint(1) 是 否 归还 








3. tb_library (图 书馆 信息 表 ) 
图 书馆 信息 表 主 要 用 于 存储 图 书馆 信息 。 该 数据 表 的 结构 如 表 6.4 所 示 。 
表 6.4 图 书馆 信息 表 结 构 






额 外 
无 AUTO INCREMENT 















记 | intao) 
libraryname | varchar(50) [utfs general ci 
varchar(10) 












utf8 general ci 
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名 字 空 

tel | varchar(20) [us general ci 是 NULL 联系 电话 
address | varchar(100) [uts general ci 是 NULL 联系 地 址 
email | varchar(100) | ua general ci 是 NULL E-mail 

wul | varchar(100) [us general ci 是 NULL 图 书馆 网 址 
createDate | date | 是 NULL 建 馆 时 间 
introduce utf8 general ci 是 图 书馆 简介 





4. tb_reader (读者 信息 表 ) 
读者 信息 表 主 要 用 于 存储 读者 的 基础 信息 。 该 数据 表 的 结构 如 表 6.5 所 示 。 
表 6.5 读者 信息 表 结构 





名 字 | 类 型 | 整理 | 空 | 默 认 | 额 外 | 说 明 
id int(10) 自动 编号 
name varchar(20) | utf8 general ci | 是 [wr| | 读者 姓名 
SEX Varchar(4) | utf8 general ci | 是 |wr| | 性 别 
barcode varchar(30) | utf8 general ci | 是 | | | 读者 条 形 码 
vocation varchar(50) | utf§ general ci | 是 [wr | | 读者 职业 
binhday | date | 是 | | | 生日 其 
paperType | varchar(10) | utf8 general ci | 是 | | | 读者 证 件 
paperNO varchar(20) | utf8 general ci | 是 | | | 证 件 号 码 
tel varchar(20) | utfg_general ci | 是 | | | 读者 电话 
email varchar(100) | utf8 general ci | 是 [wr| | E-mail 地 址 
createDate | date | 是 | | NurL | | | 办 卡 时 间 
operator varchar(30) | utf8 general ci | 是 [wr| | 操作 员 
remark mediumtext | utf8 general ci 一 一 一 备注 
typeid int(11) 读者 类 型 








人 
说明 限于 篇 幅 ， 笔 者 在 此 只 给 出 较 重要 的 数据 表 ， 其 他 数据 表 参 见 本 书 附带 的 资源 包 。 








6.5 首页 设计 











| 视频 潮解。 
651 首页 概述 
管理 员 通 过 系统 登录 模块 的 验证 后 ， 可 以 登录 到 图 书馆 管理 系统 的 首页 。 系 统 首 页 主要 包括 导航 


ea 
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栏 、 排 行 榜 和 版 权 信息 3 部 分 。 其 中 ， 导 航 栏 中 的 功能 菜单 将 根据 登录 管理 员 的 权限 进行 显示 。 例 如 ， 
系统 管理 员 mr 登录 后 ， 将 拥有 整个 系统 的 全 部 功能 ， 因 为 它 是 超级 管理 员 。 
本 案例 中 提供 的 系统 首页 如 图 6.12 所 示 。 该 页 面 在 本 书 资源 包 中 的 路 径 为 \TM\06\library\index.php。 
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图 6.12 学 校 图 书馆 管理 系统 首页 


6.5.2 首页 技术 分 析 


学 校 图 书馆 管理 系统 是 一 个 功能 全 面 、 大 型 的 Web 网 站 ， 通 过 对 网 站 的 安全 性 考虑 ， 本 网 站 对 该 
系统 进行 权限 的 分 配 ， 只 有 管理 员 级 别 的 超级 用 户 可 以 对 普通 用 户 的 权限 进行 管理 和 设置 。 系 统 首页 
主要 通过 判断 管理 员 的 权限 来 显示 该 用 户 所 操作 的 功能 模块 ， 关 键 代 码 如 下 ; 


倒 程 01 代码 位 置 ， 资源 包 \TM\06\library\navigation.php 
<?php 
include("conn/conn.php"); /| 连接 数据 库 文件 
$query=mysqli_query($conn,"select m.id,m.name,p.id,p.sysset,p.readerset,p.bookset,p.borrowback,p.sysquery from 
tb_manager as m left join (select * from tb_purview) as p on m.id=p.id where name=".$_SESSION[admin_ 
name].™); 
Sinfo=mysqli_fetch_array($query); /检索 用 户 权限 
?> 
<!-- 检 索 用 户 所 对 应 的 权限 ， 如 果 权 限 值 为 1， 则 说 明 该 功能 可 用 ， 并 输出 到 浏览 器 ， 否 则 不 显示 --> 
<td align="center"> 
<a href="index.php" class="a1"> 首 页 </a> | 

<?php if($info['sysset]==1){ ?><a onmouseover=showmenu(event,sysmenu) onmouseout=delayhidemenu() 
style="CURSOR:hand"” class="a1"> 系 统 设置 </a> | <?php}?> 
<?php if($info["readerset]==1){?><a _ onmouseover=showmenu(eventreadermenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="a1"> 读 者 管理 </a> | <?php } ?> 
<?php if($info['"bookset]==1){ ?><a href="book.php" class="a1"> 图 书 档案 管理 </a> | <?php }?> 
<?php if($info["borrowback']==1){?><a onmouseover=showmenu(event,borrowmenu) onmouseout= 
delayhidemenu() style="CURSOR:hand"class="a1" > 图 书 借 还 </a> | <?php }?> 
<?php if($info['sysquery]==1){ ?><a onmouseover=showmenu(event,querymenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="a1"> 系 统 查询 </a> | <?php } ?> 





<a href="pwd_Modify.php" class="a1"> 更 改口 令 </a> | 
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<a href="safequit.php" class="a1"> 注 销 </a> 
</td> 





of 
说明 在 权限 信息 表 tb_purview 中 ， 权 限 值 为 1， 代表 具备 该 模块 的 操作 权限 权限 值 为 0， 代 
表 不 具备 该 模块 的 操作 权限 。 


在 实现 系统 导航 菜单 时 ， 引 用 了 JavaScript 文件 menu.JS， 该 文件 中 包含 全 部 实现 半 透 明 背 景 菜单 
的 JavaScript 代码 。 


[> 技巧 将 页 面 中 所 涉及 的 JavaScript 代码 保存 在 一 个 单独 的 JS 文 件 中 ,然后 通过 <script></script> 
将 其 引用 到 需要 的 页 面 ， 可 以 规范 页 面 代 码 。 在 系统 导航 页 面 引用 menu.JS 文件 的 代码 如 下 : 


<script src="JS/menuy.JS"></script> 





6.5.3 首页 的 实现 过 程 


系统 首页 的 内 容 显示 区 用 于 显示 图 书 的 排行 信息 ， 并 将 排行 结果 按 借阅 数量 降序 排列 。 该 页 的 关 
键 代码 如 下 : 


倒 程 02 代码 位 置 ， 资源 包 \TM\06\library\index.php 





<?php 
include ("check_login.php"); // 包 含 检测 是 否 登录 的 文件 
include("conn/conn.php"); /| 连接 数据 源 文件 


$sql=mysqli_query($conn,"select * from (select bookid,count(bookid) as degree from tb_borrow group by bookid) 
as borr join (select b.*,c.name as bookcasename,p.pubname,t.typename from tb_bookinfo b left join 

tb_bookcase c on b.bookcase=c.id join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id 

where b.del=0) as book on borr.bookid=book.id order by borr.degree desc limit 10"); 


$info=mysqli_fetch_array($sql); /检索 图 书 借阅 信 息 

$i=1; 

do{ /应 用 do…while 循环 语句 显示 图 书信 息 
?> 

<tr> 


<tdalign="center"><?php echo $i;?></td> 
<td align="center"><?php echo S$info['barcode'];?></td> 
<td><?php echo S$info['bookname'];?></td> 
<td><?php echo Sinfo['typename'];?></td> 
<td align="center"><?php echo $info[bookcasename'];?></td> 
<td align="center"><?php echo S$info['pubname'];?></td> 
<td align="center"><?php echo $info['author];?></td> 
<td align="center"><?php echo S$info['price'];?></td> 
<td align="center"><?php echo S$info['degree'];?></td> 
</tr> 
<?php 


局 
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$i=$i+1; /变量 自 加 1 操作 
}while($info=mysqli_fetch_array($sql)); /do…while 循环 语句 结束 
?> 


6.6 管理 员 模 块 设 计 








6.6.1 管理 员 模 块 概述 


管理 员 模 块 主要 包括 管理 员 登 录 、 查 看 管理 员 列 表 、 添 加 管理 员 信 息 、 管 理 员 权 限 设 置 、 管 理 员 
删除 和 更 改口 令 等 6 个 功能 。 管 理 员 模块 的 框架 如 图 6.13 所 示 。 
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图 6.13 管理 员 模 块 的 框架 图 
6.6.2 ”管理 员 模 块 技 术 分 析 


在 管理 员 模 块 中 ， 涉 及 的 数据 表 是 tb_manager (管理 员 信息 表 ) 和 tb_purview〔 权 限 表 )。 其 中 ， 
管理 员 信 息 表 中 保存 的 是 管理 员 名 称 和 密码 等 信息 ， 权 限 表 中 保存 的 是 各 管理 员 的 权限 信息 ， 这 两 个 
表 通 过 各 自 的 id 字段 相关 联 。 通 过 这 两 个 表 可 以 获得 完整 的 管理 员 信息 。 


和 8 注意 在 实 现 系统 登录 前 , 需要 在 MySQL 雪 据 库 中 手动 添加 一 条 系统 管理 员 的 数据 (管理 届 名 
为 mr， 密 码 为 mrsoft， 拥 有 所 有 权限 ) ， 即 在 MySQL 的 客户 端 命令 行 中 应 用 下 面 的 语句 分 别 向 管 
理 员 信息 表 tb_manager 和 权限 表 tb_ purview 中 各 添加 一 条 数据 。 

# 添 加 管理 员 信息 

insert into tb_manager (name,pwd) values(mr,,mrsoft); 


# 添 加 权限 信息 
insert into tb_purview values(1,1,1,1,1,1); 


-一 
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从 网 站 安全 的 角度 考虑 ， 仅 仅 有 上 面 介 绍 的 系统 登录 页 面 并 不 能 有 效 地 保证 系统 的 安全 ， 一 旦 系 
统 首页 面 的 地 址 被 他 人 获得 ， 就 可 以 通过 在 地 址 栏 中 输入 系统 的 首页 面 地 址 而 直接 进入 到 系统 中 。 为 
了 便于 网 站 的 维护 ， 因 此 将 验证 用 户 是 否 登 录 的 代码 封装 在 独立 的 PHP 文件 中 ， 即 check login php 文 
件 。 验 证 用 户 是 否 登录 的 具体 代码 如 下 : 


倒 程 03 代码 位 置 资源 包 \TM\06\library\check login.php 


<?php 

session_start(); // 初 始 化 session 变量 
header("content-type:text/html;charset=utf-8"); // 设 置 页 面 编码 格式 
iflisset($_SESSION[admin_name])){ // 如 果 session 变量 为 空 , 则 说 明 用 户 未 登录 


echo "<script>alert(' 对 不 起 ， 请 通过 正确 的 途径 登录 知 源 图 书馆 管理 系统 ");window.location.href='login. 
php'</script> 
} 


?> 





当 系 统 调用 首页 时 ， 会 判断 session 变量 admin_name 是 否 存在 ， 如 果 不 存 在 ， 将 页 面 重 定向 到 系 
统 登录 (login.php) 页 面 。 


6.6.3 系统 登录 的 实现 过 程 


国 ”系统 登录 使 用 的 数据 表 : tb_manager 
系统 登录 是 进入 学 校 图 书馆 管理 系统 的 入 口 ， 主 要 用 于 验证 管理 员 的 身份 。 运 行 本 系统 ， 首 先进 
入 的 是 系统 登录 页 面 ， 在 该 页 面 中 ， 系 统管 理 员 可 以 通 
过 输入 正确 的 管理 员 名 称 和 密码 登录 到 系统 首页 ， 当 用 





户 没有 输入 管理 员 名 称 或 密码 时 ， 系 统 会 通过 JavaScript 全 exe 于 
进行 判断 ， 并 给 予 信息 提示 。 系 统 登录 页 面 的 运行 结果 = 
如 图 6.14 所 示 。 a sa| 0 
系统 登录 页 面 主要 用 于 收集 管理 员 的 输入 信息 及 通 
过 自 定义 的 JavaSeript 函数 验证 输入 信息 是 否 为 空 .该 页 i 
面 中 所 涉及 的 表单 元 素 如 表 6.6 所 示 。 
表 6.6 系统 登录 页 面 所 涉及 的 表单 元 素 









重要 属性 





含义 

















forml | form method="post" action="chklogin.php" | 管理 员 登 录 表 单 
name | text size="25" | 管理 员 名 称 
pwd | password size="25" | 管理 员 密 码 
submit | value=" 确 定 " onclick="retum check(form1)"” | 。 “确定 ”按钮 
submit3 | value=" 重 置 " | “ 重 置 ” 按钮 





value=" 关 闭 " onClick="window.closeO:" “关闭 ”按钮 


编写 自 定 义 的 JavaScript 函数 ， 用 于 判断 管理 员 名 称 和 密码 是 否 为 空 。 代 码 如 下 : 


全 
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倒 程 04 ”代码 位 置 : 资源 包 \TM\06\library\login.php 





<script language="javascript"> 
function check(form}{ // 自 定义 一 个 JavaScript 函数 check() 
if (form.name.value=="™")}{ // 如 果 管理 员 名 称 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 管理 员 名 称 "");form.name .focus();return false; 


} 
if (form.pwd.value=="™"){ // 如 果 管 理 员 密码 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 密码 "");form.pwd.focus();return false; 
} 
} 
</script> 





提交 表单 到 数据 处 理 页 ， 页 面 中 为 了 防止 非法 用 户 进入 学 校 图 书馆 管理 系统 首页 ， 通 过 调用 类 的 
chkinput0 方 法 实现 判断 用 户 名 和 密码 是 否 正确 。 如 果 为 合法 用 户 ， 则 可 以 登录 学 校 图 书馆 管理 系统 的 
首页 否则， 弹出 相应 的 错误 提示 。 关 键 代 码 如 下 : 


倒 程 05 代码 位 置 ， 资源 包 \TM\06\library\chklogin.php 





<?php 
session_start(); /初始 化 session 变量 
header("content-type:text/html;charset=utf-8"); /| 设置 页 面 编码 格式 
$A_name=$_POST['name']; /接收 表单 提交 的 用 户 名 
$A_pwd=$_POST[pwd']; /接收 表单 提交 的 密码 
@ classchkinput{ /定义 类 
var $name; 
var $pwd; 
function __construct ($x,$y){ /定义 一 个 方法 
S$this->name=$x; // 将 管理 员 名 称 传 给 类 对 象 $this->name 
Sthis->pwd=$y; // 将 管理 员 密 码 传 给 类 对 象 $this->pwd 
} 
function checkinput(){ 
include("conn/conn.php"); // 连 接 数据 库 文 件 
$sql=mysqli_query($conn,"select * from 了 b_manager where name=".$this->name." and pwd=". $this->pwd.™"); 
$info=mysqli_fetch_array($sql); /检索 管理 员 名 称 和 密码 是 否 正确 
if($info==falseX{ // 如 果 管 理 员 名 称 或 密码 不 正确 ， 则 弹出 相关 提示 信息 
echo "<scriptlanguage=javascript>alert( 您 输入 的 管理 员 名 称 或 密码 错误 ， 请 重新 输入 ! '); history.back(); 
</script>"; 
exit; 
} 
else{ // 如 果 管 理 员 名 称 或 密码 正确 ， 则 弹出 相关 提示 信息 
echo "<script>alert(' 管 理 员 登录 成 功 !);window.location='index.php';</script>"; 
$_SESSION[admin_name]=$inforname']; // 将 管理 员 名 称 存 到 $_SESSION[admin_name'] 变 量 中 
$_SESSION[pwd]=$inforpwd]: /将 管理 员 密码 存 到 $_SESSION['pwd] 变 量 中 
} 
} 
} 
© $obj=new chkinput(trim($name),trim($pwd)); // 创 建 对 象 
©  $obj->checkinput(); /1 调用 类 
?> 
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代码 贴 二 

@ class: 创建 一 个 PHP 类 时 ， 必 须 使 用 关键 字 class 进行 声明 ， 该 关键 字 后 紧 跟 类 的 名 称 ， 之 后 用 大 括号 将 类 体 进 
行 封装 。 

@ new: 在 PHP 中 应 用 new 关键 字 创 建 对 象 。 该 模块 创建 了 一 个 名 为 chkinput 的 验证 管理 员 类 。 

@ $obj->checkinput0:: 通过 $obj 对 象 调用 checkinput 类 中 的 属性 和 方法 ， 即 管理 员 名 称 和 密码 。 


6.6.4 查看 管理 员 的 实现 过 程 


国 查看 管理 员 使 用 的 数据 表 : tb manager、tb_purview 

管理 员 登 录 后 ， 选 择 “ 系 统 设置 ”/“ 管 理 员 设 置 ”菜单 项 ， 进 入 到 查看 管理 员 列 表 页 面 。 在 该 页 
面 中 ， 将 以 表格 的 形式 显示 全 部 管理 员 
及 其 权限 信息 , 并 提供 添加 管理 员 信息 、 
删除 管理 员 信息 和 设置 管理 员 权 限 的 超 
链接 。 查 看 管理 员 列表 页 面 的 运行 结果 让 


如 图 6.15 所 示 。 a 系 近 于 证 管理 加 村 尘世 从 汪 。 系 要 坦 询 相 限 训 下 操作 


首先 使 用 左 外 联接 语句 (left join… 区 二 
on) 从 数据 表 tb_manager 和 tb_purview 已 nt 
中 查询 出 符合 条 件 的 数据 ， 然 后 将 查询 a rmt 
结果 应 用 do…while 循环 语句 输出 到 浏 图 6.15 查看 管理 员 列 表 页 面 的 运行 结果 


览 器 。 关 键 代码 如 下 : 


倒 程 06 代码 位 置 , 资源 包 \TM\02\library\manager.php 

<?php 

include("conn/conn.php"); // 连 接 数据 库 文件 

© $sql=mysqli_query($conn,"select m.id,m.name,p.sysset,p.readerset,p.bookset,p.borrowback,p.sysquery 
from tb_manager as m left join (select * from tb_purview) as p on m.id=p.id"); 










Saw 图 书 谊 管理 系统 


如 年， 不 生物 芝 








Sinfo=mysqli_fetch_array($sql); // 检 索 数 据 信息 

do{ /应 用 do…while 循环 语句 输出 查询 结果 
?> 

roseoororretenetrsosts 输 出 符合 查询 条 件 的 记录 rrrrrerrrrsrrrrsrrrrrrrrorrers 4 

<tr> 


<td style="padding:5px;"><?php echo $info['name'];?></td> 
四 <tdalign="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" 
disabled="disabled" <?php if($info['sysset]==1){fecho ("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" 
disabled="disabled" <?php if($info['readerset"]==1){echo("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info['bookset]==1){echo("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info["borrowback'"]==1){echo("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info['sysquery"]==1)}{echo("checked");}?>></td> 

<td align="center"><a href="#" onClick="window.open('manager_modify.php?id=<?php echo 

局 
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$infof'id]; ?>',",'width=340,height=215)"> 权 限 设置 </a></td> 
<td align="center"><a href="manager_del.php?id=<?php echo $info["id"];?>"> 删 除 </a></td> 
</tr> 
/人 
<?php 
}while($info=mysqli_fetch_array($sql)); /ldo…while 循环 语句 结束 
?> 





< 代码 贴 二 
@ left join…on: 应 用 左 外 联接 将 两 个 表 或 多 个 表 连 接 起 来 ， 返 回 部 分 或 全 部 匹配 行 ， 详 解 参见 6.10.2 节 。 
@ <?php if($info['sysset']=—1) {echo ("checked"):}?>: 如 果 系 统 设置 字段 的 值 为 1， 则 复 选 框 处 于 选中 状态 。 


6.6.5 ”添加 管理 员 的 实现 过 程 


国 添加 管理 员 使 用 的 数据 表 : tb manager 

管理 员 登 录 后 ， 选 择 “ 系 统 设置 ”/“ 管 理 员 设置 ”菜单 项 ， 
进入 到 查看 管理 员 列 表 页 面 ， 在 该 页 面 中 单 击 “ 添 加 管理 员 信 
息 ” 超 链接 ， 打 开 添 加 管理 员 信息 页 面 。 添 加 管理 员 信息 页 面 
的 运行 结果 如 图 6.16 所 示 。 





和 6 注意 泊 泊 加 的 管理 员 信 息 没有 权限 ， 必 须 通 过 设置 和 二 = — 
理 员 权限 为 其 指定 可 操作 的 功能 模块 。 图 6.16 添加 管理 员 信息 页 面 的 运行 结果 
在 查看 管理 员 列 表 页 面 ， 单 击 “ 添 加 管理 员 信息 ” 超 链接 文字 的 HTML 代码 如 下 ; 


倒 程 07 代码 位 置 ， 资源 包 \TM\06\library\manager.php 
<a href="#" onClick="window.open('manager_add.php',","width=340,height=215)"> 添 加 管理 员 信息 </a> 


添加 管理 员 页 面 主要 用 于 收集 输入 的 管理 员 信息 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 
否 合 法 。 该 页 面 中 所 涉及 的 表单 元 素 如 表 6.7 所 示 。 








表 6.7 添加 管理 员 页 面 所 涉及 的 表单 元 素 


























名 称 元 素 类 型 重要 属性 

forml form method="post" action="manager_ok.php" | 表单 

name text | 管理 员 名 称 
pwd password | 管理 员 密 码 
pwdl password | 确认 密码 
submit submit value=" 保 存 " onClick="return check(form1)" | “保存 ”按钮 












“关闭 ”按钮 


在 添加 管理 员 页 面 中 ， 输 入 合法 的 管理 员 名 称 及 密码 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 表 单 信息 到 数 
据 处 理 页 ， 将 添加 的 管理 员 信息 保存 到 数据 表 中 。 如 果 添加 成 功 ， 弹 出 成 功 的 提示 信息 ; 否则 ， 弹 出 


_ 国 


submit2 button 





value=" 关 闭 " onClick="window.closeO:" 
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错误 提示 。 代 码 如 下 : 


倒 程 08 代码 位 置 资源 包 \TM\02\library\manager_ok.php 


<?php 

header("content-type:text/html;charset=utf-8"); // 设 置 页 面 编码 格式 

include("conn/conn.php"); /| 连接 数据 库 文 件 

if($_POST[submit]!="™"){ /如果 单 击 了 “保存 ”按钮 ， 则 执行 下 面 的 操作 
$name=$_POST[name']; // 获 取 管 理 员 名 称 

S$pwd=$_POST[pwd']; // 获 取 管 理 员 密码 
$sql=mysqli_query($conn,"insert into tb_manager (name,pwd) values('$name','$pwd')"); 

if($sql==true){ // 向 数据 表 中 添加 管理 员 信息 成 功 ， 则 给 出 提示 信息 


echo "<script language=javascript>alert(' 管 理 员 添加 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; 

} 

else{ /向 数据 表 中 添加 管理 员 信息 失败 ， 则 给 出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 添加 失败 ! ');window.close();window.opener.location.reload(); 
</script>"; 

} 

} 





| 


6.6.6 ”设置 管理 员 权限 的 实现 过 程 


链接 ， 即 可 进入 到 “权限 设置 ”页 面 ， 设 置 该 管理 员 的 操作 权限 。 


只 0 技巧 在 添加 管理 员 处 理 页 中 应 用 “window.opener.location.reload();” 语 名 刷新 父 窗口 中 的 信息 。 








[iceshor /TWOo minay /meron le 
| meeanenrwosm 





管理 员 名 和 


拥有 的 权限 ， 凤 图 书 管理 日 图 书 借 还 


回 设置 管理 员 权限 使 用 的 数据 表 : tb manager、tb purview 
在 查看 管理 员 列表 页 面 单 击 指定 管理 员 后 面 的 “权限 设置 ” 超 












| 其 


权限 设置 页 面 的 运行 结果 如 图 6.17 所 示 。 6.17 权限 设置 页 面 的 运行 结果 


权限 设置 页 面 中 所 涉及 的 表单 元 素 如 表 6.8 所 示 。 
表 6.8 ”权限 设置 页 面 所 涉及 的 表单 元 素 









































名 称 元 素 类 型 重要 属性 含义 
forml form method="post" action="manager modifyokphp” 表单 
id hidden value="<?php echo $info["id]:?>" 管理 员 编号 
name text value="<?php echo $info['name']:?>" 管理 员 名 称 
Sysset checkbox <?php if($info['sysset =—1){echo("checked"):}?> 系统 设置 
Teaderset checkbox <?php if($info['readerset |]=—1) {echo("checked"):}?> 读者 管理 
bookset checkbox <?php if($info['bookset]=—1) {echo("checked"):}?> 图 书 管理 
borrowback checkbox <?php if($info['borowback']=—1){echo("checked"):}?> 图 书 借 还 
Sysquery, Checkbox <?php if($info['sysquery']=—=1){echo("checked"):}?> 系统 查询 
submit submit class="btn grey" “保存 ”按钮 
Submit2 button value=" 关 闭 " onClick="window.closeO:" “关闭 ”按钮 
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在 查看 管理 员 列表 页 面 中 ， 添 加 权限 设置 列 ， 并 在 该 列 中 添加 以 下 用 于 打开 权限 设置 页 面 的 超 链 
接 代码 。 


倒 程 09 代码 位 置 资源 包 \TM\06\library\manager.php 
<a href="#" onClick="window.open('manager_modify.php?id=<?php echo S$info['id"]; ?>',","width=340,height= 
215)"> 权 限 设置 </a> 


从 上 面 的 URL 地 址 中 可 以 获取 设置 管理 员 权 限 页 所 涉及 的 id 号 ,将 id 号 提交 给 处 理 页 manager_ 
modifyok.php， 修 改 id 号 所 对 应 的 管理 员 信息 。 具 体 代 码 如 下 : 


倒 程 10 ”代码 位 置 ， 资源 包 \TM\06\library\manager_modifyok.php 





<?php 

header("content-type:text/html;charset=utf-8"); // 设 置 页 面 编码 格式 

include("conn/conn.php"); /连接 数据 库 文件 

if($_POST[submit]!=") // 如 果 提 交 表 单 ， 则 执行 以 下 操作 
$id=$_POST[id]; // 获 取 id 信 息 

@ $sysset=!lisset($_POST['sysset"])?0:1; /应 用 三 目 运算 符 求 出 系统 设置 ” 复 选 框 的 值 


$readerset=lisset($_POST['readerset])?0:1; 
$bookset=lisset($_POST[bookset'])?0:1; 
$borrowback=lisset($_POST["borrowback'])?0:1; 





$sysquery=lisset($_POST['sysquery])?0:1; /应 用 三 目 运算 符 求 出 “ 系统 查询 ” 复 选 框 的 值 
$query=mysqli_query($conn,"select * from tb_purview where id=$id"); 

S$info=mysqli_fetch_array($query); // 检 索 权限 信息 表 中 是 否 存在 该 管理 员 
if($info==false){ // 如 果 不 存在 ， 向 权限 表 中 添加 管理 员 权限 信息 


@ mysqliquery($conn,"insert into tb_purview(id,sysset,readerset,bookset,borrowback,sysquery) 
values($id,$sysset,$readerset, $bookset, $borrowback, $sysquery)"); 


} 
else{ // 否 则 ， 更 新 管理 员 的 权限 信息 
mysqli_query($conn,"update tb_purview set 
Sysset=$sysset,readerset=$readerset,bookset=$bookset,borrowback=$borrowback,sysquery=$sysquery 
where id="$id"); 


} 

echo"<script language=javascript>alert(' 权限 设置 修改 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; // 更 新 成 功 ， 弹 出 提示 信息 ， 并 更 新 父 窗口 

} 


?> 





人 代码 贴 二 
@S$_POST['sysset] 一 ""?0:1:: 应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 ， 如 果 等 于 空 ， 值 为 0， 否则 值 为 1。 
@ insert…into: 向 权限 信息 表 中 添加 一 行 数据 信息 ，insert…into 语句 只 适用 于 对 单行 数据 的 插入 。 


6.6.7 ”删除 管理 员 的 实现 过 程 


国 删除 管理 员 使 用 的 数据 表 : tb manager、tb_purview 
在 查看 管理 员 列表 页 面 中 ， 单 击 指定 管理 员 信息 后 面 的 “删除 ” 超 链接 ， 该 管理 员 及 其 权限 信息 
将 被 删除 。 
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在 查看 管理 员 列表 页 面 中 添加 以 下 用 于 删除 管理 员 信息 的 超 链接 代码 。 


倒 程 11 代码 位 置 资源 包 \TM\06\library\manager.php 
<a href="manager_del.php?id=<?php echo $info['id1];?>"> 删 除 </a> 





从 上 面 的 URL 地 址 中 ， 可 以 获取 删除 管理 员 所 涉及 的 id 号 ,将 id 号 提交 给 manager_del.php 处 理 
页 删除 id 号 所 对 应 的 管理 员 信息 。 具 体 代码 如 下 : 


倒 程 12 代码 位 置 ， 资源 包 \TM\06\library\manager_del.php 


<?php 

header("content-type:text/html:charset=utf-8"); // 设 置 页 面 编码 格式 
include("conn/conn.php"); // 连 接 数 据 库 文件 
$id=$_GET[id]; /获取 管理 员 的 id 号 


$sql=mysqli_query($conn,"delete from tb_manager where id='$id"); /删除 管理 员 表 中 id 号 所 对 应 的 管理 员 信息 

$query=mysqli_query($conn,"delete from tb_purview where id='$id"); /删除 权限 表 中 id 号 所 对 应 的 管理 员 权限 

if($sql==true and $query==true}{ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 删除 成 功 ! ');history.back();</script>"; 


} 

else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 删除 失败 ! ');history.back();</script>"; 

} 


?> 





6.6.8 单元 测试 


在 开发 完 管理 员 模块 后 ， 需 要 对 该 模块 进行 单元 测试 。 当 管理 员 修改 操作 员 的 权限 时 ， 如 果 未 赋 
予 该 操作 员 所 有 权限 ， 那 么 在 单 击 “ 保 存 ” 按 钮 后 ， 该 操作 员 的 权限 并 没有 改变 。 下 面 给 出 修改 权限 
设置 处 理 页 的 源 代码 : 





<?php 

header("content-type:text/html;charset=utf-8"); /设置 页 面 编码 格式 
include("conn/conn.php"); // 连 接 数 据 库 文件 
if($_POST['submit]!=™"X /如 果 提 交 表 单 ， 则 执行 以 下 操作 
$id=$_POST[id]; /获取 id 信息 
$sysset=$_POST['sysset]; 1/ 获取 “系统 设置 ” 复 选 框 的 值 
$readerset=$_POST[readerset]; /获取 “ 读 者 管理 ” 复 选 框 的 值 
$bookset=$_POST['bookset]; /获取 “ 图 书 管理 ” 复 选 框 的 值 
$borrowback=$_POST['borrowback']; /获取 “图 书 借 还 ” 复 选 框 的 值 
$sysquery=$_POST['sysquery]; /获取 “系统 查询 ” 复 选 框 的 值 
$query=mysqli_query($conn,"select * from tb_purview where id=$id"); 

Sinfo=mysqli_fetch_array($query); // 检 索 权限 信息 表 中 是 否 存在 该 管理 员 
if($info==false){ // 如 果 不 存在 ， 向 权限 表 中 添加 管理 员 权限 信息 


mysqli_query($conn,"insert into tb_purview(id,sysset,readerset,bookset,borrowback,sysquery) 
values(S$id,$sysset, $readerset, $bookset, $borrowback, $sysquery)"); 
} 
else{ // 否 则 ， 更 新 管理 员 的 权限 信息 


@ 
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mysqli_query($conn,"update tb_purview set 
Sysset=$sysset,readerset=$readerset,bookset=$bookset,borrowback=$borrowback,sysquery=$sysquery 
where id="$id"); 
下 
echo"<script language=javascript>alert( 权限 设置 修改 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; // 更 新 成 功 ， 弹 出 提示 信息 ， 并 更 新 父 窗口 
于 


称 





为 了 找 出 错误 的 原因 ， 笔 者 应 用 echo0 语 句 对 $sysset 等 一 系列 传递 的 值 进行 输出 ， 输 出 的 结果 显 
示 ， 当 未 选中 某 个 权限 时 ， 页 面 中 会 输出 错误 提示 ， 这 充分 说 明 在 未 选中 某 个 复 选 框 时 ， 没 有 获取 到 
该 复 选 框 的 值 。 

解决 该 问题 的 方法 是 对 提交 的 复 选 框 的 状态 进行 判断 ， 如 果 复 选 框 处 于 未 选中 状态 ， 则 值 为 0; 如 
果 处 于 选中 状态 ， 则 值 为 1。 这 样 就 可 以 获取 到 复 选 框 状态 的 值 。 

处 理 复 选 框 的 状态 有 两 种 方法 : 一 种 是 应 用 让 条 件 语句 对 逐个 值 进行 判断 ， 判 断 复 选 框 的 当前 状 
态 ， 并 赋予 其 值 (但 这 种 方法 过 于 烦琐 )， 另 一 种 是 应 用 三 目 运算 符 ， 简 单 快捷 地 计算 复 选 框 的 值 (这 
种 方法 可 以 简化 代码 ， 本 模块 采用 该 方法 获取 复 选 框 的 值 )。 

应 用 三 目 运算 符 获 取 复 选 框 的 值 的 代码 如 下 : 





$sysset=lisset($_POST['sysset])?0:1; /应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 
$readerset=lisset($_POST[readerset])?0:1; /应 用 三 目 运算 符 求 出 “读者 管理 ” 复 选 框 的 值 
$bookset=lisset($_POST["bookset])?0:1; /应 用 三 目 运算 符 求 出 “图 书 管理 ” 复 选 框 的 值 
$borrowback=lisset($_POST['borrowback'])?0:1; /应 用 三 目 运算 符 求 出 “图 书 借 还 ” 复 选 框 的 值 
$sysquery=lisset($_POST['sysquery'])?0:1; /应 用 三 目 运算 符 求 出 “系统 查询 ” 复 选 框 的 值 





完整 的 权限 设置 处 理 页 的 代码 参见 例 程 10。 


6.7 读者 档案 管理 模块 设计 








6.7.1 读者 档案 管理 模块 概述 


读者 档案 管理 模块 主要 包括 查看 读者 列表 、 添 加 读者 信息 、 修 改 读者 信息 、 删 除 读者 信息 和 查看 
读者 详细 信息 等 5 个 功能 。 读 者 档案 模块 的 框架 如 图 6.18 所 示 。 


读者 信息 列表 


f 


查看 读者 列表 添加 读者 信息 修改 读者 信息 删除 读者 信息 查看 读者 详细 信息 


_ 国 
















































































图 6.18 读者 档案 模块 的 框架 图 
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6.7.2 ”读者 档案 管理 模块 技术 分 析 


在 读者 档案 管理 模块 中 , 涉及 的 数据 表 是 tb_reader( 读 者 信息 表 ) 和 tb_readertype (读者 类 型 信息 
表 )， 其 中 ， 读 者 信息 表 中 保存 的 是 读者 的 姓名 、 条 形 码 和 读者 类 型 等 信息 ， 读 者 类 型 信息 表 中 保存 的 
是 读者 的 类 型 ， 这 两 个 数据 表 间 通过 读者 类 型 id 字段 进行 关联 。 通 过 这 两 个 数据 表 可 以 获得 完整 的 读 
者 档案 信息 。 对 读者 详细 信息 进行 查询 的 SQL 语句 如 下 : 

$sql=mysqli_query($conn,"select ridrbarcode,rname,rsex,tname as typename,r.vocation,r.birthday,r.paperType, 


r.paperNO ,r.tel,r.email,r.remark from tb_reader as rjoin (select * from tb_readertype) as t on r.typeid=t.id where 
rid=".$_GET[id].""); 


S$ 和 6 注意 本 异 次 主要 应 用 联接 语句 实现 多 表 坦 询 ， 关 于 联接 二 的 详细 讲解 参见 611 节 ， 


6.7.3 查看 读者 信息 列表 的 实现 过 程 


国 。 查看 读者 信息 列表 使 用 的 数据 表 : tb_reader、tb_readertype 

管理 员 登 录 后 ， 选 择 “ 读 者 管理 ”/ ~ 
“读者 档案 管理 ”菜单 项 , 进入 到 查看 
读者 列表 页 面 , 在 该 页 面 中 将 显示 全 部 
读者 信息 , 同时 提供 添加 读者 信息 、 修 
改 读者 信息 和 删除 读者 信息 的 超 链 接 。 
查看 读者 信息 列表 页 面 的 运行 结果 如 
图 6.19 所 示 。 

在 查询 全 部 读者 信息 时 , 首先 应 用 
join…on 内 联接 语句 将 tb_reader 和 
tb_readertype 数据 表 连接 起 来 , 然后 应 
用 do…while 语句 循环 输出 查询 结果 。 

















查看 读者 信息 列表 页 面 的 代码 如 下 : 图 6.19 查看 读者 信息 列表 的 运行 结果 
倒 程 13 ”代码 位 置 ， 资源 包 \TM\06\library\reader.php 
<?php 
include("conn/conn.php"); /J 连接 数据 库 文件 


$sql=mysqli_query($conn,"select rid,rbarcode,rname,tname as typename,r.paperType,r.paperNO,r.tel,r.email 
from tb_reader as r join (select * from tb_readertype) as t on r.typeid=t.id"); 


$info=mysqli_fetch_array($sql); /应 用 内 联接 检索 读者 信息 
?> 
二 /省略 读 者 信息 标题 HTML 标记 部 分 
<?php 
do{ /应 用 do…while 循环 语句 输出 查询 结果 
?> 
<tr> 


<td><?php echo $info[barcode]:?> </td> 


SS 
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<td><a href="reader_info.php?id=<?php echo $infor'id]; ?> "><?php echo Sinfo[name'];?> </a></td> 
<td><?php echo S$info[typename'];?> </td> 
<td align="center"><?php echo Sinfo[ paperType'];?> </td> 
<td><?php echo $info[paperNO];?> </td> 
<td align="center"><?php echo S$info['tel];?> </td> 
<td><?php echo $info[email];?> </td> 
<td width="6%" align="center"><a href="reader_modify.php?id=<?php echo $info['id1;?>"> 修 改 </a></td> 
<td width="5%" align="center"><a href="reader_del.php?id=<?php echo $info['id1;?> "> 删除 </a></td> 
</tr> 
<?php 
}while($info=mysqli_fetch_array($sql)); /ldo…while 循环 语句 结束 
?> 


6.7.4 查看 读者 详细 信息 的 实现 过 程 


回 查看 读者 详细 信息 使 用 的 数据 表 : tb_reader、tb_readertype 
管理 员 登 录 后 ， 在 读者 列表 页 面 中 单 击 读者 姓名 的 超 链接 ， 可 以 进入 到 查看 指定 读者 详细 信息 的 
页 面 。 在 该 页 面 中 将 显示 该 读者 的 详细 信息 。 查 看 读者 详细 信息 页 面 的 运行 结果 如 图 6.20 所 示 。 
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图 6.20 查看 读者 详细 信息 的 运行 结果 


在 查询 读者 详细 信息 时 ， 以 URL 中 传递 的 读者 id 作为 查询 条 件 ， 应 用 join…on 内 联接 语句 对 
tb_reader 和 tb_readertype 数据 表 进行 查询 。 查 看 读者 详细 信息 页 面 的 代码 如 下 : 


倒 程 14 代码 位 置 资源 包 \TM\06\library\reader_info.php 
<?php 
include("conn/conn.php"); /连接 数据 库 文件 
$sql=mysqli_query($conn,"select r.id,r.barcode,r.name,r.sex,t.name as typename,r.vocation,r.birthday,r. 
paperType,r.paperNO,r.tel,r.email,r.remark from tb_reader as r join (select * from tb_readertype) as t on 
r.typeid=t.id where r.id=".$_GET[id].™); /定义 内 联接 语句 
$info=mysqli_fetch_array($sql); /检索 读者 信息 
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> 
<table width="600" height="432" border="0" cellpadding="0" cellspacing="0" bgcolor="#FFFFFF"> 
<tr> 
<td width="173" align="center"> 姓 名 : </td> 
<td width="427" height="39"> 
<input name="name" type="text" value="<?php echo $info[name'];?>"> 
</td> 
</tr> 
<tr> 
<td width="173" align="center"> 性 别 : </td> 
<td height="35"> 
<?php 
if($info['sex'"]==" 男 "}{ 
?> 
<input name="sex" type="radio" class="noborder" id="radiobutton” value=" 男 " checked> 男 
<?php 
jelsef 
?> 
<input name="sex" type="radio" class="noborder" value=" 女 " checked> 女 
<?php}?> 
</td> 
</tr> 
<tr> 
<td align="center"> 条 形 码 : </td> 
<td><input name="barcode" type="text" id="barcode" value="<?php echo S$info['barcode'];?>"> 
</td> 
</tr> 
<tr> 
<td align="center"> 读 者 类 型 :</td> 
<td><input name="typename" type="text" id="typename" value="<?php echo S$info[typename'];?>"> 
</td> 
</tr> 
<tr> 
<td align="center"> 职 业 : </td> 
<td><input name="vocation" type="text" id="vocation" value="<?php echo S$info['vocation'];?>"></td> 
</tr> 
<tr> 
<td align="center"> 出 生日 期 : </td> 
<td><input name="birthday" type="text" id="birthday" value="<?php echo Sinfo['birthday'];?>"></td> 
</tr> 
<tr> 
<td align="center"> 有 效 证 件 : </td> 
<td><input name="paperType" type="text" id="paperType" value="<?php echo $info[paperType];?>"> 
</td> 
</tr> 
<tr> 


<td align="center"> 证 件 号 码 : </td> 
<td><input name="paperNO" type="text" id="paperNO" value="<?php echo $info[ paperNO'];?>"> 
</td> 
</tr> 
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<tr> 
<td align="center"> 电 话 : </td> 
<td><input name="tel" type="text" id="tel" value="<?php echo S$info['tel];?>"></td> 
</tr> 
<tr> 
<td align="center">E-mail: </td> 
<td><input name="email" type="text" id="email" value="<?php echo $info[email];?>" size="50"> 
</td> 
</tr> 
<tr> 
<td align="center"> 备 注 : </td> 
<td><textarea name="remark" cols="60" rows="6" class="wenbenkuang" id="remark"><?php echo 
Sinfo['remark'];?></textarea></td> 
</tr> 


<tr> 
<td align="center">&nbsp;</td> 
<td> 
<input name="Submit2" type="button" class="btn_grey" value=" 返 回 " onClick="history.back()"></td></tr> 
</table> 


6.7.5 添加 读者 信息 的 实现 过 程 


回 添加 读者 信息 使 用 的 数据 表 : tb reader、tb readertype 
管理 员 登 录 系 统 后 ， 在 读者 列表 页 面 中 单 击 “添加 读者 信息 ” 超 链接 ， 进 入 到 添加 读者 信息 页 面 。 
添加 读者 信息 页 面 的 运行 结果 如 图 6.21 所 示 。 
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图 6.21 添加 读者 信息 页 面 的 运行 结果 
在 查看 读者 列表 页 面 中 ， 设 置 “ 添 加 读者 信息 ” 超 链 接 的 代码 如 下 : 


倒 程 15 代码 位 置 资源 包 \TM\06\library\reader.php 
<a href="reader_add.php"> 添 加 读者 信息 </a> 





PHP 项 目 开发 全 程 实录 (第 4 版 ) 


添加 读者 信息 页 面 主要 用 于 收集 输入 的 读者 信息 , 并 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 
否 合 法 。 该 页 面 中 所 涉及 的 重要 表单 元 素 如 表 6.9 所 示 。 


表 6.9 添加 读者 信息 页 面 所 涉及 的 重要 表单 元 素 




















名 称 元 素 类 型 重要 属性 贫 ” 义 
forml form method="post" action="reader ok.php" 表单 
barcode text id="barcode" 读者 条 形 码 
<?php include("conn/conn.php"); 
$sql=mysqli_query($conn,"select * 位 om 
tb_readertype"); 
Sinfo=mysqli fetch array($sq)); 
typeid select dof 读者 类 型 
?> 
<option value="<?php echo $info['id"]:?>"> 
<?php echo $info[mame']:?></option> 
<?php}while($info=mysqli fetch array($sql)):?> 
<option value=" 身 份 证 " selected> 身 份 证 </option> 
tion value=" 学 生 证 "> 学 生 证 </option> 
PS ype Gi ee value=" 军 官 证 "> 军 定 钙 opp 有 效 证 件 
<option value=" 工 作证 "> 工作 证 </option> 
operator hidden value="<?php echo $ SESSION['admin name']:?>" 操作 员 
Submit submit onClick="retum check(forml)" “保存 ”按钮 
Submit2 button onClick="historybackO:" “返回 ”按钮 


在 添加 读者 信息 页 面 的 表单 中 输入 读者 信息 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 表 单 信息 到 数据 处 理 页 ， 
将 添加 的 读者 信息 保存 到 数据 表 中 。 如 果 添 加 成 功 ， 弹 出 成 功 的 提示 信息 ; 否则 ， 弹 出 失败 的 提示 信 
息 。 代 码 如 下 : 


倒 程 16 代码 位 置 ， 资源 包 \TM\06\library\reader_ok.php 


<?php 

header("content-type:text/html;charset=utf-8"); // 设 置 文件 编码 格式 

include("conn/conn.php"); // 连 接 数据 库 文件 

date_default_timezone_set("PRC"); /设置 时 区 

if($_POST[Submit]!="™")}{ // 如 果 单 击 了 “保存 ”按钮 
S$name=$_POST[name]; /获取 读 者 姓名 
S$sex=$_POST['sex']; 1/ 获 取 读 者 性 别 
S$barcode=$_POST[barcode']; // 获 取 读 者 条 形 码 
$typeid=$_POST[typeid]; // 获 取 读 者 类 型 id 
S$vocation=$_POST[vocation]; /获取 读者 职业 
$birthday=$_POST[birthday]; /获取 读 者 出 生日 期 
S$paperType=$_POST[paperType']; // 获 取 读 者 证 件 类 型 
S$paperNO=$_POST[paperNO"; 1/ 获 取 读 者 证 件 号 码 
Stel=$_POSTTtel]; /获取 读者 电话 
$email=$_POST[email]: /获取 读者 邮箱 地 址 


274 


第 6 章 学 校 图 书馆 管理 系统 ( Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 





ScreateDate=date("Y-m-d"); /获取 当前 日 期 
$operator=$_POST[operator]; /获取 操作 员 
Sremark=$_POST[remark"]; /获取 备注 信息 


$sql=mysqli_query($conn,"insert into tb_reader (name,sex,barcode,typeid,vocation,birthday,paperType, 
paperNO ,tel,email,createDate,operator,remark) 
values('$name','$sex','$barcode','$typeid','$vocation','$birthday','$paperType','$paperNO','$tel','$email','$create 
Date','$operator','$remark")"); /定义 并 执行 插入 语句 
if$sql==true)f 
echo "<script language=javascript>alert( 读 者 信息 添加 成 功 ! ');history.back();location.href='reader.php'; 
</script>"; 
} 
else{ 
echo "<script language=javascript>alert(' 读 者 信息 添加 失败 ! ');history.back();window.opener.location. 
reload();</script>"; 


} 


?> 


6.7.6 ”修改 读者 信息 的 实现 过 程 


国 修改 读者 信息 使 用 的 数据 表 : tb_reader、tb_readertype 
管理 员 登 录 系 统 后 ， 在 读者 列表 页 面 中 单 击 读者 信息 后 面 的 “修改 ” 超 链接 ， 进 入 到 “修改 读者 
信息 ”页 面 。 修 改 读 者 信息 页 面 的 运行 结果 如 图 6.22 所 示 。 





当前 位置 ， 训 管庄 从 村 萤 理 》 从 于 江 委 信息 

















图 6.22 ”修改 读者 信息 页 面 的 运行 结果 
在 读者 信息 列表 页 面 中 ， 添 加 “修改 ” 超 链接 的 代码 如 下 : 


倒 程 17 代码 位 置 资源 包 \TM\06\library\reader.php 
<a href="reader_modify.php?id=<?php echo $info['id];?>"> 修 改 </a> 
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在 修改 读者 信息 页 面 中 修改 读者 信息 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 表 单 信息 到 数据 处 理 页 reader_ 
modifyok.php， 应 用 UPDATE 语句 将 修改 的 读者 信息 保存 到 数据 表 tb_reader 中 ， 并 弹出 “读者 信息 修 
改 成 功 !” 的 提示 信息 ， 将 页 面 重 定向 到 读者 信息 列表 页 。 数 据 处 理 页 的 代码 如 下 : 


倒 程 18 ”代码 位 置 资源 包 \TM\06\library\reader_modifyok.php 


<?php 

session_start(); /初始 化 session 变量 
header("content-type:text/html;charset=utf-8"); // 设 置 页 面 编码 格式 
include("conn/conn.php"); // 连 接 数据 库 文件 
$id=$_POST[readerid]; // 获 取 读 者 id 
$name=$_POST[name'; // 获 取 读 者 姓名 
$sex=$_POST['sex]; // 获 取 读 者 性 别 
$barcode=$_POST[barcode]; // 获 取 读 者 条 形 码 
$vocation=$_POST[vocation1]; // 获 取 读 者 职业 
S$birthday=$_POST[birthday"]; // 获 取 读 者 出 生日 期 
S$paperType=$_POST[paperType']; // 获 取 读 者 证 件 类 型 
$paperNO=$_POST[paperNO1]; // 获 取 读 者 证 件 号 码 
Stel=$_POST[tel]; // 获 取 读 者 电话 
$email=$_POST[email]; // 获 取 读 者 邮箱 地 址 
$remark=$_POST[remark]; // 获 取 读 者 备注 信息 
S$typeid=$_POST[typeid"]; // 获 取 读 者 类 型 d 


mysqli_query($conn,"update tb_reader set name='$name',sex='$sex',barcode='"$barcode',vocation='$vocation', 
birthday="$birthday',paperType='$paperType',paperNO='$paperNO,",tel='S$tel',email='$email',remark="$remark',ty 
peid='$typeid' where id="$id"); 

echo "<script language='javascript'>alert(' 读 者 信息 修改 成 功 !");window.location.href='reader.php';</script>"; 

?> 





6.7.7 ”删除 读者 信息 的 实现 过 程 


国 ”删除 读者 信息 使 用 的 数据 表 : tb_reader、tb_readertype 

管理 员 登 录 系 统 后 ， 在 读者 列表 页 面 中 单 击 读者 信息 后 面 的 “删除 ” 超 链接 ， 即 可 把 指定 的 读者 

信息 删除 。 例 如 ， 删 除 姓名 为 “叮当 ”的 读者 后 的 运行 结果 如 图 6.23 所 示 。 
ee 






人 














了 


图 6.23 ”删除 读者 信息 后 的 运行 结果 
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在 查看 读者 列表 页 面 中 ， 设 置 “ 删 除 ” 超 链接 的 代码 如 下 : 


倒 程 19 代码 位 置 资源 包 \TM\06\library\reader.php 
<a href="reader_del.php?id=<?php echo $info[id];?> "> 删除 </a> 


单 击 想 要 删除 的 读者 信息 后 面 的 “删除 ” 超 链接 ， 提 交 表单 信息 到 数据 处 理 页 reader_del.php， 应 
用 DELETE 语句 将 指定 的 读者 信息 从 数据 表 tb_reader 中 删除 ， 如 果 删 除 操作 执行 成 功 ， 则 弹出 “读者 
信息 删除 成 功 !” 的 提示 信息 ， 并 将 页 面 重 定向 到 读者 信息 列表 页 面 。 数 据 处 理 页 的 代码 如 下 : 


倒 程 20 ”代码 位 置 : 资源 包 \TM\06\library\reader_del.php 


<?php 

header("content-type:text/html;charset=utf-8"); /1 设置 页 面 编码 格式 
include("conn/conn.php"); /| 连接 数据 库 文件 
$id=$_GET[id]; /获取 删除 读者 信息 的 id 
$sql=mysqli_query($conn,"delete from tb_reader where id='$id"); /| 执行 删除 语句 

if($sql){ // 如 果 信 息 删 除 成 功 ， 则 弹出 提示 


echo "<script language=javascript>alert(' 读 者 信息 删除 成 功 !'");window.location.href='reader.php';</script>"; 


else{ 
echo "<script language=javascript>alert(' 读 者 信息 删除 失败 !');window.location.href='reader.php';</script>"; 
} 


?> 





6.8 图 书 档案 管理 模块 设计 





6.8.1 图 书 档案 管理 模块 概述 


图 书 档案 管理 模块 主要 包括 查看 图 书 列表 、 添 加 图 书信 息 、 修 改 图 书信 息 、 删 除 图 书信 息 和 查看 
图 书 详细 信息 等 5 个 功能 。 图 书 档案 模块 的 框架 如 图 6.24 所 示 。 














图 书信 息 列 表 
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查看 图 书 列表 添加 图 书信 息 修改 图 书信 息 删除 图 书信 息 查看 图 书 详细 信息 

































































图 6.24 图 书 档案 模块 的 框架 图 
6.8.2 ”图 书 档案 管理 模块 技术 分 析 


在 图 书 档 案 管 理 模块 中 ， 涉 及 的 数据 表 是 tb_bookinfo (图 书信 息 表 )、tb_bookcase (书架 设置 表 )、 
tb_booktype《〈 图 书 类 型 表 ) 和 tb_publishing〈 出 版 社 信 息 表 )， 这 4 个 数据 表 间 通过 相应 的 字段 进行 关 


Ea 
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联 , 如 图 6.25 所 示 。 通过 以 上 4 个 表 可 以 获得 完整 的 图 
书 档案 信息 。 























6 注 总 相模 类 主 妥 应 用 联接 语 各 实现 多 表 查 询 ， 
关于 联接 语句 的 详细 讲解 参见 6.11 节 。 fmt Ae 
6.8.3 ”查看 图 书信 息 列表 的 实现 过 程 Ei 
图 6.25 图 书 档案 管理 模块 各 表 间 关系 图 

























tor varchar (30) 
al tinyint (1) 



























国 ”查看 图 书信 息 列表 使 用 的 数据 表 : tb bookinfo 、 
tb_bookcase、 tb_booktype、 tb_publishing 

管理 员 登 录 后 ， 在 导航 栏 中 选择 “图 书 档案 管理 ”菜单 项 ， 进 入 到 查看 图 书 列表 页 面 ， 在 该 页 面 
中 将 显示 全 部 图 书信 息 列表 ， 同 时 提供 添加 图 书信 息 、 删 除 图 书信 息 、 修 改 图 书信 息 的 超 链接 。 查 看 
图 书信 息 列表 页 面 的 运行 结果 如 图 6.26 所 示 。 
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6.26 查看 图 书信 息 列表 的 运行 结果 
打开 功能 导航 navigation.php 文件 ， 设 置 “ 图 书 档案 管理 ”菜单 项 的 超 链 接 的 代码 如 下 : 


倒 程 21 代码 位 置 ， 资 源 包 \TMWOGibrarymavigation .php 
<a href="book.php" class="a1"> 图 书 档案 管理 </a> 











首先 应 用 join…on 内 联接 语句 将 tb_bookinfo、tb_bookcase、tb_booktype 和 tb_publishing 4 个 数据 
表 连 接 起 来 检索 指定 条 件 的 图 书信 息 , 然后 应 用 do…while 循环 语句 输出 查询 结果 到 浏览 器 。 查 看 图 书 
信息 页 面 的 代码 如 下 : 


倒 程 22 ”代码 位 置 : 资源 包 \TM\06\library\book.php 

<?php 

include("conn/conn.php"); /连接 数据 库 文件 

$query=mysqli_query($conn,"select book.barcode,book.id as bookid,book.bookname,bt.typename,pb.pubname, 
bc. name from tb_bookinfo book join tb_booktype bt on book.typeid=bt.id join tb_publishing pb on book.ISBN= 
pb.ISBN join tb_bookcase bc on book.bookcase=bc.id"); 


全 
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Sresult=mysqli_fetch_array($query); /应 用 外 联接 检索 图 书信 息 
?> 
// 省 略图 书信 息 标题 HTML 标记 部 分 
<?php 
do{ /应 用 do…while 循环 语句 输出 查询 结果 
?> 
<tr> 


<td><?php echo $resultfbarcode'];?></td> 
<td><a href="book_look.php?id=<?php echo $resultrbookid];?>"><?php echo 
$result[rbookname'];?></a></td> 
<td><?php echo $result[typename];?></td> 
<td><?php echo $resultrpubname'];?></td> 
<td><?php echo $result[name'];?></td> 
<td align="center"><a href="book_Modify.php?id=<?php echo $result[bookid1];?>"> 修 改 </a></td> 
<td align="center"><a href="book_del.php?id=<?php echo $resultrbookid];?>"> 删 除 </a></td> 
</tr> 
<? 
}while($result=mysqli_fetch_array($query)); /ldo…while 循环 语句 结束 
?> 


入 注意 关于 join…on 内 联接 语句 的 使 用 方法 参见 6.11.1 节 。 


6.8.4 添加 图 书信 息 的 实现 过 程 


回 添加 图 书信 息 使 用 的 数据 表 : tb_bookinfo、tb_bookcase、tb_booktype、tb_publishing 

管理 员 登 录 系统 后 ， 在 导航 栏 中 单 击 “ 图 书 档案 管理 ” 超 链接 ， 进 入 到 查看 图 书 列表 页 面 。 在 该 
页 面 中 单 击 “ 添 加 图 书信 息 ” 超 链接 ， 进 入 到 添加 图 书信 息 页 面 。 添 加 图 书信 息 页 面 的 运行 结果 如 
图 6.27 所 示 。 





3 


WE] ee 





[TPR ET 
坟 尖 二 人骨 role 测 二 路 er1050 为 量 仁 星 示 光 昌 


图 6.27 添加 图 书信 息 页 面 的 运行 结果 
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在 查看 图 书 列表 页 面 中 ， 设 置 “ 添 加 图 书信 息 ” 超 链接 的 代码 如 下 : 


倒 程 23 ”代码 位 置 : 资源 包 \TM\06\library\book.php 
<a href="book_add.php"> 添 加 图 书信 息 </a> 





添加 图 书信 息 页 面 主要 用 于 收集 输入 的 图 书信 息 以 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 
是 否 合法 。 该 页 面 中 所 涉及 的 重要 表单 元 素 如 表 6.10 所 示 。 


名 称 


forml 


typeld 


isbn 


bookcaseid 


表 6.10 ”添加 图 书信 息 页 面 所 涉及 的 重要 表单 元 素 


元 素 类 型 重要 属性 富 ” 义 
form method="post" action="book okphp" 表单 
<?php include("conn/conn.php"); 图 书 类 型 


$sql=mysqli query($conn."select * from tb booktype"): 


Select 


Sinfo=mysqli fetch array($sq)):; 
do{ 

?> 

<option value="<?php echo $info['id]:?>"> 国人 

<?php echo $info['typename'];?></option> 

<?php} while($info=mysqli fetch array($sq)):?> 

<?php 
$sql2=mysqli_query($conn,"select * from tb_publishing"); 
Sinfo2=mysqli fetch_array($sql2); 


ot 出 版 社 


select 
> 


<option value="<?php echo $info2[ISBN']:?>"> 
<?php echo $info2[pubname']:?></option> 
<?php} while($info2=mysqli fetch array($sql2)):?> 
<2php 
$sql3=mysqli_query($conn,"select * from tb_bookcase"):; 
S$info3=mysqli_fetch array($sql3): 
dof 
> 
<option value="<?php echo $info3['id]:?>"> 
<?php echo $info3['name']:?></option> 
<?php} while($info3=mysqli fetch array($sql3)):?> 


书架 名 称 


select 





Operator 


hidden value="<?php echo $info3['name']:?>" 操作 员 





Submit 


submit ‘onClick="return check(form1)" “保存 ”按钮 





Submit2 





button onClick="history.backO:" “返回 ”按钮 


由 于 添加 图 书信 息 的 方法 同 添加 管理 员 信息 的 方法 类 似 ， 所 以 此 处 只 给 出 向 图 书信 息 表 中 插入 数 





据 的 SQL 语句 


， 详 细 代码 参见 资源 包 。 向 图 书信 息 表 中 插入 数据 的 SQL 语句 如 下 : 
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倒 竹 24 ”代码 位 置 : 资源 包 \TM\06\library\book_ok.php 

mysqli_query($conn,"insert into 
tb_bookinfo(barcode,bookName,typeid,authortranslatorISBN,price,page,bookcase,inTime,operator )values('$b 
arcode','$bookName','$typeid','$author','$translator','$isbn','$price','$page','$bookcaseid','$inTime','$operator’)"); 





6.8.5 修改 图 书信 息 的 实现 过 程 


国 修改 图 书信 息 使 用 的 数据 表 : tb_bookinfo、tb_bookcase、tb_booktype、tb_publishing 

管理 员 登 录 系统 后 ， 在 导航 栏 中 单 击 “ 图 书 档案 管理 ” 超 链接 ， 进 入 到 查看 图 书 列表 页 面 。 单 击 
想 要 修改 的 图 书信 息 后 面 的 “修改 ” 超 链接 ， 进 入 到 “修改 图 书信 息 ” 页 面 。 修 改 图 书信 息 页 面 的 运 
行 结果 如 图 6.28 所 示 。 








Er 
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图 6.28 ”修改 图 书信 息 页 面 的 运行 结果 
在 图 书信 息 列表 页 面 中 ， 添 加 “修改 ” 超 链接 的 代码 如 下 : 
倒 程 25 ”代码 位 置 : 资源 包 \TM\06\library\book.php 
<a href="book_Modify.php?id=<?php echo $result[bookid];?>"> 修 改 </a> 


在 修改 图 书信 息 页 面 中 修改 图 书信 息 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 表 单 信息 到 数据 处 理 页 book_ 
Modify ok.php， 应 用 UPDATE 语句 将 修改 的 图 书信 息 保存 到 数据 表 tb_bookinfo 中 ， 并 弹出 “图 书信 
息 修 改 成 功 ! ”提示 信息 ， 将 页 面 重 定向 到 修改 图 书信 息 页 。 数 据 处 理 页 的 代码 如 下 : 


倒 程 26 代码 位 置 资源 包 \TM\06\library\book_Modify_ok.php 














<?php 


session_start(); /| 初始 化 session 变量 
header("content-type:text/html;charset=utf-8"); /1 设置 页 面 编码 格式 
include("conn/conn.php"); // 连 接 数 据 库 文件 
date_default_timezone_set("PRC"); ll 设置 时 区 
S$bid=$_POST[bid']: // 获 取 图 书 id 号 
$operator=$_SESSION[admin_name]; /1 获取 管理 员 名 称 


_ 团 
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$barcode=$_POST[barcode'; 1/ 获取 图 书 条 形 码 
S$bookName=$_POST[bookName'; 1/ 获取 图 书 名 称 
Stypeid=$_POSTTtypeld"]; // 获 取 图 书 类 型 id 号 
$author=$_POST[author]; /获取 图 书 作者 
$translator=$_POST[translator]; // 获 取 图 书 译 者 
S$isbn=$_POST['isbn']; 1/ 获取 出 版 社 ISBN 
$price=$_POST[price]; /获取 图 书 单价 
$page=$_POST[page]; /获取 图 书页 码 
$bookcase=$_POST[bookcaseid1]; // 获 取 图 书 书架 id 号 
SinTime=date("Y-m-d"); // 设 置 图 书 更 新 日 期 为 当前 日 期 


$query=mysqli_query($conn,"update tb_bookinfo set barcode='$barcode', bookName='$bookName' , typeid='$typeid', 
author='$author, translator='$translator, ISBN='$isbn' , price='$price’ , page='$page' , bookcase='$bookcaseid', 


inTime='$inTime', operator='$operator where id=$bid"); /更 新 数据 表 
echo "<script language='javascript>alert(' 图 书信 息 修改 成 功 !);history.back();</script>"; 
?> 


6.8.6 ”删除 图 书信 息 的 实现 过 程 
国 删除 图 书信 息 使 用 的 数据 表 : tb_bookinfo 
在 查看 图 书 列表 页 面 中 ， 设 置 “ 删 除 ” 超 链接 的 代码 如 下 : 


倒 程 27 ”代码 位 置 ， 资源 包 \TM\06\library\book.php 
<a href="book_del.php?id=<?php echo $result[bookid];?>"> 删 除 </a> 








单 击 想 要 删除 的 图 书信 息 后 面 的 “删除 ” 超 链 接 ， 提 交 表 单 信息 到 数据 处 理 页 book_delphp， 应 用 
DELETE 语句 将 指定 的 图 书信 息 从 数据 表 tb_bookinfo 中 删除 ， 如 果 删 除 操作 执行 成 功 ， 则 弹出 “图 书 
信息 删除 成 功 !” 提 示 信 息 ， 并 将 页 面 重 定向 到 图 书信 息 列 表 页 面 。 数 据 处 理 页 的 代码 如 下 ; 


倒 程 28 ”代码 位 置 ， 资源 包 \TM\06\library\book_del.php 





<?php 

include("conn/conn.php"); /连接 数据 库 文件 
header("content-type:text/html;charset=utf-8"); /设置 页 面 编码 格式 
Sinfo_del=mysqli_query($conn,"delete from tb_bookinfo where id=".$_GET[id]); /删除 指定 的 图 书信 息 
if($info_del){ // 如 果 信息 删除 成 功 , 则 弹出 提示 


echo "<script language='javascript'>alert(' 图 书信 息 删 除 成 功 !");history.back();</script> "; 








6.9 图 书 借 还 模块 设计 


1 视频 讲解 
6.9.1 图 书 借 还 模块 概述 
图 书 借 还 模块 主要 包括 图 书 借阅 、 图 书 续 借 、 图 书 归还 、 图 书 档案 查询 、 图 书 借阅 查询 、 借 阅 到 


282 


第 6 章 学 校 图 书馆 管理 系统 ( Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


期 提醒 等 6 个 功能 。 在 图 书 借阅 模块 中 的 用 户 只 有 一 种 身份 ， 那 就 是 操作 员 ， 通 过 该 身份 可 以 进行 图 
书 借 还 等 相关 操作 。 图 书 借 还 模块 的 用 例 图 如 图 6.29 所 示 。 


EE 
4 
图 书 借阅 查询 > < 一 人 一 ><C 图 书 续 借 > 
1 商人 只 Ac 而 HE 
图 6.29 图 书 借 还 模块 的 用 例 图 


6.9.2 图 书 借 还 模块 技术 分 析 


在 图 书 借 还 模块 中 涉及 的 数据 表 是 tb_borow〈 图 书 借阅 信息 表 )、tb_bookinfo (图书 信息 表 )〉 和 
tb_reader (读者 信息 表 )， 这 3 个 数据 表 间 通过 相应 的 字段 进行 关联 ， 如 图 6.30 所 示 。 














book hborrow 
加 iteger wasigned Ck) 
resderid integer unsigned 《BC》 

er (10) 《ad 








date 
warchar GD) 
ifbask tinyint (1) 




















operator 。 warehar (30) 


varcher toxt 
del tinyint (1) typeid integer (11) 











图 6.30 图书 借 还 管理 模块 各 表 间 关系 图 
6 注意 二 模 关 主 要 应 用 联接 滞 习 实现 多 表 查 询 ， 关 于 联接 语 各 的 详细 讲解 参见 6.11 闻 。 


6.9.3 图 书 借阅 的 实现 过 程 


国 图书 借 阅 使 用 的 数据 表 : tb_borrow、tb_bookinfo、tb reader 

管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 借 阅 ” 菜 单项 ， 进 入 到 图 书 借 阅 页 面 ， 在 该 页 面 的 “ 读 
者 条 形 码 ”文本 框 中 输入 读者 的 条 形 码 〈 如 556699) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 读 
者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 将 其 显示 在 页 面 中 ， 此 时 输入 
图 书 的 条 形 码 或 图 书 名 称 后 ， 单 击 “ 确 定 ”按钮 ， 借 阅 指定 的 图 书 ， 运 行 结果 如 图 6.31 所 示 。 


py 
3 昌 当 读者 借阅 图 书 完毕 后 ， 操 作 员 通 过 单 击 “ 完 成 借阅 ”按钮 ， 将 重新 载 入 图 书 借阅 页 面 ， 
当前 页 处 于 空 信息 状态 ， 从 而 方便 操作 员 对 下 一 个 读者 进行 借阅 图 书 操作 。 


图 书 借阅 页 面 总 体 上 可 以 分 为 两 个 部 分 : 一 部 分 用 于 查询 并 显示 读者 信息 ; 另 一 部 分 用 于 显示 读 
者 的 借阅 信息 和 添加 读者 借阅 信息 。 
@ 
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图 6.31 图书 借阅 页 面 的 运行 结果 


在 进行 图 书 借阅 时 ， 系 统 要 求 每 个 读者 只 能 同时 借阅 一 定数 量 的 图 书 ， 并 且 该 数量 由 读者 类 型 表 
tb_readertype 中 的 可 借 数量 number 决定 ， 所 以 笔者 编写 了 自 定义 的 checkbookO 函 数 ， 用 于 判断 当前 选 
择 的 读者 是 否 还 可 以 借阅 新 的 图 书 ， 同 时 该 函数 还 具有 判断 输入 读者 条 形 码 或 图 书 名 称 文本 框 是 否 为 
空 的 功能 。 代 码 如 下 : 


倒 程 29 ”代码 位 置 ， 资源 包 \TM\06\bookBorrow.php 
<script language="javascript"> 
function checkbook(form}X{ // 自 定义 一 个 JavaScript 函数 checkbook() 
if(form.barcode.value=="™" // 如 果 读 者 条 形 码 为 空 
alert(" 请 输入 读者 条 形 码 "");form.barcode.focus();return; // 弹 出 提示 ， 焦 点 返回 到 条 形 码 文本 框 


} 
if(form.inputkey.value=="™" // 如 果 图 书 查询 文本 框 的 值 为 空 
alert(" 请 输入 查询 关键 字 ");form.inputkey.focus();return; /弹出 提示 ， 焦 点 返回 到 图 书 查询 文本 杠 


} 
ifform.numbervaluel=" && form.numbervalue-form.borrowNumbervalue<=0){ “// 如 果 图 书 的 借阅 数量 超 


过 了 可 借 数量 
alert(" 您 不 能 再 借阅 其 他 图 书 了 !");return; // 弹 出 提示 信息 
} 
form.submit(); /提交 表单 
} 
</script> 


> 


[2 在 JavaScript 中 比较 两 个 数值 型 文本 框 的 值 时 ， 不 使 用 运算 符 “--”， 而 是 将 这 两 个 值 相 
减 ， 再 判断 其 结果 。 


检索 读者 的 基本 信息 的 SQL 语句 如 下 : 


© 
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倒 程 30 ”代码 位 置 资源 包 \TM\06\library\bookborrow.php 


$sql=mysqli_query($conn,"select r.*,t.name as typename,t.number from tb_reader r left join tb_readerType t on 
r.typeid=t.id where rbarcode=".$barcode.""); 
S$info=mysqli_fetch_array($sql); /检索 读者 信息 


获取 读者 借阅 信息 的 SQL 语句 如 下 : 


倒 程 31 代码 位 置 ， 资源 包 \TM\06\library\bookborrow.php 


$sql1=mysqli_query($conn,"select r*,borrborrowTime,borrbackTime,book.bookname,book.price,pub.pubname,bc. 
name as bookcase from tb_borrow as borr join tb_bookinfo as book on book.id=borr bookid join tb_publishing as 
pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id join tb_reader as r on 
borrreaderid= rid where borrreaderid='$readerid' and borr.ifoack=0"); 

$info1=mysqli_fetch_array($sql1); /检索 读者 的 借阅 信息 
$borrowNumber=mysqli_num_rows($sql1); /获取 结果 集中 行 的 数目 


在 “图 书 条 形 码 ”/“ 图 书 名 称 ”文本 框 中 输入 图 书 条 形 码 或 图 书 名 称 后 ， 单 击 “ 确 定 ”按钮 ， 检 
索 该 图 书 在 图 书 借阅 信息 表 中 是 否 存在 ， 如 果 不 存在 ， 则 向 图 书 借 阅 信息 表 中 添加 该 读者 的 图 书 的 借 
阅 记录 ， 完 成 图 书 借阅 操作 ， 和 否则 ， 弹 出 该 书 不 能 被 同一 读者 重复 借阅 的 提示 信息 。 图 书 借阅 的 具体 
代码 如 下 : 


倒 程 32 代码 位 置 ， 资源 包 \TM\06\library\bookBorrow.php 











<?php 
if($info && isset($_POST[inputkey]) && $_POSTTinputkey]!="){// 如 果 “ 图 书 条 形 码 ”/“ 图 书 名 称 ”文本 框 不 为 空 
$f=$_POST[f]; // 获 取 用 户 选 择 的 条 件 值 
@ $inputkey=trim($_POST[inputkey"]); 1/ 获取 用 户 输入 的 查询 关键 字 
S$barcode=$_POST[barcode']; 1/ 获取 读 者 的 条 形 码 
@ $borrowTime=date('Y-m-d"); // 图 书 的 借阅 时 间 为 系统 当前 时 间 
© $backTime=date("Y-m-d",(time()+3600*24*30)); // 归 还 图 书 日 期 为 当前 期 日 期 +30 天 期 限 
$query=mysqli_query($conn,"select * from tb_bookinfo where $f='S$inputkey"); 
Sresult=mysqli_fetch_array($query); /检索 图 书信 息 是 否 存在 
说 $result==false){ // 如 果 读 者 借阅 的 图 书 不 存在 ， 那 么 弹出 提示 信息 


echo "<script language=javascript>alert( 该 图 书 不 存在 ! 
');window.location.href='bookBorrow.php?barcode=$barcode'; </script>"; 


} 
else{ /检索 该 读者 所 借阅 的 图 书 是 否 与 再 借 图 书 重复 
$query1=mysqli_query($conn,"select r.*,bor.borrowTime,bor.backTime,book.bookname,book.price, pub.pubname, 
bc. name as bookcase from tb_borrow as borr join tb_reader as r on borr.readerid=r.id join tb_bookinfo as book 
on book.id=borr.bookid join tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book. 
bookcase=bc.id where borr.bookid=$result[id] and borrreaderid=$readerid and ifback=0"); 
$result1=mysqli_fetch_array($query1); 
if($result1==trueX{ // 如 果 所 借 图 书 已 被 该 读者 借阅 ， 那 么 提示 不 能 重复 借阅 
echo "<script language=javascript>alert( 该 图 书 已 经 借阅 ! ')windowlocation.href=,bookBorrowphp? 
barcode=$barcode';</script>"; 
4 
else{ // 否 则 ， 完 成 图 书 借阅 操作 ， 并 弹出 借阅 成 功 提示 信息 
S$bookid=S$result[id"]; // 将 读者 id 号 赋 给 一 变量 
mysqli_query($conn,"insert into 
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tb_borrow(readerid,bookid,borrowTime,backTime,operator,ifback)values('$readerid','$bookid','$borrowTime','$back 
Time'",".$_SESSION[admin_name'].",0)"); /向 借阅 信息 表 中 添加 一 条 借阅 信息 
echo "<script language='javascript>alert(' 图 书 借阅 操作 成 功 ! ' 
);window.location.href='bookBorrow.php?barcode=$barcode';</script>"; 
: 
} 
} 


Vy 





< 代码 贴 十 

@trim0: 删除 字符 串 中 首尾 的 空白 或 者 其 他 字符 ， 以 达到 精确 查询 。 

@ date(Y-m-d): 获取 系统 的 当前 日 期 为 图 书 借 阅 的 日 期 ， 并 格式 化 日 期 格式 。 

@ date("Ym-d",(time0+3600*24*30)): 归还 图 书 日 期 = 当前 期 日 期 130 天 期 限 。 当前 日 期 时 间 戳 应 用 time0 函 数 获取 ， 
30 天 期 限 的 时 间 蕉 等 于 3600 秒 x24 小 时 x30 天 ， 并 通过 date() 函 数 格式 化 为 指定 日 期 格式 。 


6.9.4 图 书 续 借 的 实现 过 程 


国 。 图书 续 借 使 用 的 数据 表 : tb borrow、tb_ bookinfo、tb reader 

管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 续 借 ”菜单 项 ， 进 入 到 图 书 续 借 页 面 。 在 该 页 面 的 “ 读 
者 条 形 码 ”文本 框 中 输入 读者 的 条 形 码 〈 如 556699) 后 ， 单 击 “ 确 定 ”按钮 ， 系 统 会 自动 检索 出 该 读 
者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 则 将 其 显示 在 页 面 中 ， 此 时 单 
击 “ 续 借 ” 超 链接 ， 即 可 续 借 指定 图 书 〈 即 将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 )。 


ol 
所 培 明 当 读者 续 借 完 图 书后 ， 操 作 员 通 过 单 击 “ 完 成 续 借 ”按钮 ， 将 重新 载 入 图 书 续 借 页 面 ， 
当前 页 处 于 空 信息 状态 ， 从 而 方便 操作 员 进行 下 一 个 读者 续 借 图 书 操作 。 


图 书 续 借 页 面 的 设计 方法 同 图 书 借阅 类 似 ， 所 不 同 的 是 ， 在 图 书 续 借 页 面 中 没有 添加 借阅 图 书 的 
功能 ， 而 是 添加 了 “ 续 借 ” 超 链 接 。 图 书 续 借 页 面 的 运行 结果 如 图 6.32 所 示 。 
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图 6.32 图 书 续 借 页 面 的 运行 结果 
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单 击 “ 续 借 ” 超 链接 时 ， 还 需要 将 读者 条 形 码 、 借 阅 id 号 和 图 书 归 还 时 间 一 同 传递 到 图 书 续 借 的 
处 理 页 borrow_oncemore php 中 。 代 码 如 下 : 


倒 程 33 代码 位 置 资源 包 \TM\06\library\bookRenew.php 


<a href="borrow_oncemore.php?barcode=<?php echo $info['barcode'];?>&borrid=<?php echo 
Sinfo['borrid];?>&backTime=<?php echo $info[backTime1];?>"> 续 借 </a> 


检索 读者 信息 和 读者 借阅 信息 的 SQL 语句 如 下 : 


倒 程 34 代码 位 置 ， 资源 包 \TM\06\library\bookRenew.php 


$sql=mysqli_query($conn,"select borrid as borrid,borrborrowTime,borrbackTime,borrifback,r*tname as 
typename, tnumberbook.bookname,book.price,pub.pubname,bc.name as bookcase from tb_borrow as borr 
join tb_reader r on borr.readerid=r.id join tb_readerType t on r.typeid=t.id join tb_bookinfo as book on book.id= 
borr.bookid join tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase= 
bc.id where rbarcode=".$barcode." and borr.ifoack=0"); 

$info=mysqli_fetch_array($sql); /检索 读者 信息 和 借阅 信息 


单 击 “ 续 借 ” 超 链接 ， 提 交 到 数据 处 理 页 borrow_oncemore.php， 主 要 用 于 完成 图 书 的 续 借 功能 ， 
主要 通过 更 改 图 书 的 归还 日 期 (即将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 ， 续 借 日 期 
的 具体 算法 参见 6.10.1 节 的 详细 讲解 ) 实现 。 数 据 处 理 页 的 代码 如 下 ; 


倒 程 35 ”代码 位 置 ， 资源 包 \TM\06\library\borrow_oncemore.php 











<?php 

session_start(); /初始 化 session 变量 
header("content-type:text/html;charset=utf-8"); // 设 置 页 面 编码 格式 
include("conn/conn.php"); /| 连接 数据 库 文件 
S$barcode=$_GET[barcode']; // 获 取 图 书 条 形 码 
Snew=$_GET[backTime']:; // 获 取 图 书 归 还 时 间 
date_default_timezone_set("PRC"); // 设 置 时 区 


// 更 新 续 借 期 ， 将 动态 获取 的 还 书 日 期 转化 为 时 间 戳 ， 然 后 再 求 出 续 借 后 的 还 书 日 期 
SnewbackTime=date("Y-m-d",(mktime(0, 0, 0, substr($new 5,2), substr($new 8,2), substr($new,0,4))+3600*24*30)); 
S$borrid=$_GETTborrid"]; // 获 取 续 借 图 书 的 id 号 
mysqli_query($conn,"update tb_borrow set backTime='$newbackTime ',ifback=0,operator=".$_SESSION[admin_ 
name']." where id=$borrid"); 

echo "<script language=javascript>alert(' 图 书 续 借 操作 成 功 ! ');window.location.href='bookRenew.php? barcode= 
$barcode'; </script> // 弹 出 图 书 续 借 成 功 的 提示 信息 


?> 


6.9.5 图 书 归 还 的 实现 过 程 


国 。 图书 归 还 使 用 的 数据 表 : tb_borrow、tb_bookinfo、tb_reader 

管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 归 还 ”菜单 项 ， 进 入 到 图 书 归 还 页 面 。 在 该 页 面 的 “ 读 
者 条 形 码 ” 文 本 框 中 输入 读者 的 条 形 码 〈 如 556699) 后 ， 单 击 “确定 ”按钮 ， 系 统 会 自动 检索 出 该 读 
者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 则 将 其 输出 到 浏览 器 ， 此 时 单 
击 “ 归 还 ” 超 链接 ， 即 可 将 指定 图 书 归 还 。 图 书 归 还 页 面 的 运行 结果 如 图 6.33 所 示 。 
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图 6.33 图书 归还 页 面 的 运行 结果 


图 书 归还 页 面 的 设计 方法 同 图 书 续 借 类 似 ， 所 不 同 的 是 ， 将 图 书 续 借 页 面 中 的 “ 续 借 ” 超 链接 更 
换 为 “归还 ” 超 链 接 。 在 单 击 “ 归 还 ” 超 链 接 时 ， 也 需要 将 读者 条 形 码 和 借阅 id 号 一 同 传递 到 图 书 归 
还 处 理 页 。 代 码 如 下 : 


倒 程 36 ”代码 位 置 ， 资源 包 \TM\06\library\bookBack.php 
<a href="bookBack_ok.php?borrid=<?php echo $info[borrid];?>&barcode=<?php echo $info['barcode'];?>"> 归 
还 </a> 








检索 读者 信息 及 读者 借阅 信息 的 SQL 语句 如 下 : 


倒 程 37 代码 位 置 ， 资源 包 \TM\06\library\bookBack.php 


$sql=mysqli_query($conn,"select borrid as borrid,borrborrowTime,borrbackTime,borrifback,r*tname as 
typename, tnumberbook.bookname,book.price,pub.pubname,bc.name as bookcase from tb_borrow as borr 
join tb_reader r on borrreaderid=rid join tb_readerType t on r.typeid=t.id join tb_bookinfo as book on book.id= 
borr.bookid join tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase= 
bc.id where rbarcode=".$barcode." and borrifback=0"); 

$info=mysqli_fetch_array($sql); /检索 读者 信息 及 该 读者 的 借阅 信息 


单 击 “ 归 还 ” 超 链接 ， 即 可 将 指定 图 书 归还 。 数 据 处 理 页 的 代码 如 下 : 
倒 程 38 ”代码 位 置 ， 资源 包 \TM\06\library\bookBack_ok.php 


<?php 

session_start(); /初始 化 session 变量 
header("content-type:text/html;charset=utf-8"); /设置 页 面 编码 格式 
include("conn/conn.php"); /连接 数据 库 文件 
date_default_timezone_set("PRC"); /设置 时 区 

$backTime=date("Y-m-d"); /归还 图 书 日 期 

S$borrid=$_GET[borrid']; // 获 取 读 者 的 id 号 
mysqli_query($conn,"update tb_borrow set backTime='$backTime',ifback=1,operator=".$_SESSION[admin_ 
name']." where id=$borrid"); /更 新 读者 的 借阅 信息 


局 


第 6 章 学 校 图 书馆 管理 系统 ( Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 





echo "<script language='javascript>alert(' 图 书 归还 操作 成 功 ! ');window.location.href='bookBack.php?barcode= 
$barcode';</script>"; 弹出 图 书 归还 成 功 的 提示 信息 


Ec 





6.9.6 图 书 借阅 查询 的 实现 过 程 


国 图书 借 阅 查询 使 用 的 数据 表 : tb borrow、tb bookinfo、tb reader 

管理 员 登 录 后 ， 选 择 “系统 查询 ”/“ 图 书 借阅 查询 ”菜单 项 ， 进 入 到 图 书 借阅 查询 页 面 。 图 书 借 
阅 查 询 页 面 的 运行 结果 如 图 6.34 所 示 。 在 该 页 面 中 可 以 按 指定 的 字段 或 某 一 时 间 段 进行 查询 ， 同 时 还 
可 以 实现 按 指定 字段 及 时 间 段 进行 综合 条 件 查询 。 






当前 们 党， 图 书 曾 区 》 醒 书信 阅 查 匀 


za-09 已 还 











6.34 ”图 书 借阅 查询 页 面 的 运行 结果 


图 书 借阅 查询 页 面 主要 用 于 收集 查询 条 件 和 显示 查询 结果 , 并 通过 自 定义 的 JavaScript 函数 验证 输 
入 的 查询 条 件 是 否 合法 。 该 页 面 中 所 涉及 的 表单 元 素 如 表 6.11 所 示 。 


表 6.11 ”图书 借阅 查询 页 面 所 涉及 的 表单 元 素 














名 称 元 素 类 型 重要 属性 含义 
myform form method="post" action="" 表单 
flagl checkbox value="a" 请 选择 查询 依据 
flag2 checkbox value="b" 借阅 时 间 
<option value="k.barcode" > 图 书 条 形 码 </option> 
tion value="k.bookname"> </option> 
上 sleet se ee gp 查询 字段 
<option value="r.name"> 读 者 名 称 </option> 
keyl text Size="50" 关键 字 
sdate text id="sdate" 开始 日 期 
edate text id="edate" 结束 日 期 
Submit submit ‘onClick="return check(myform):" “查询 ”按钮 











值 ， 
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在 图 书 借阅 查询 页 面 中 ， 指 定 查询 条 件 后 ， 提 交 表 单 信息 到 当前 页 。 首 先 获取 表单 元 素 复 选 框 的 
然后 根据 复 选 框 的 值 组 合 查 询 字符 串 。 
如 果 flagl 的 值 等 于 a， 那么 按 指 定 的 字段 检索 图 书 借阅 信息 ; 如 果 flag2 的 值 等 于 b， 那 么 按 指 定 


的 时 间 段 检索 图 书 借阅 信息 ; 如 果 hlagl 的 值 等 于 a， 并且 flag2 的 值 等 于 b， 那 么 按 以 上 两 个 条 件 的 综 
合 条 件 检索 图 书 借阅 信息 ， 并 将 查询 结果 输出 到 浏览 器 。 具 体 代码 如 下 : 


290 


倒 程 39 代码 位 置 资源 包 \TM\06\library\borrowQuery.php 

<?php 

include("conn/conn.php"); /| 连接 数据 库 文件 
$sql=mysqli_query($conn,"select ”bborowTime,bbackTime,bjifbackrbarcode as readerbarcode,rname,Kkiid,k.barcode, 
Kk.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id"); /查询 
图 书 借阅 信息 


iflisset($_POST['Submit])X{ // 如 果 提交 了 表单 ， 则 执行 以 下 操作 
$f=$_POSTIf]; // 获 取 操 作 员 选择 的 查询 条 件 
$key1=$_POST[key11]; // 获 取 查 询 关 键 字 
$sdate=$_POST[sdate]; // 获 取 借 阅 的 起 始 日 期 
$edate=$_POST[edate]; // 获 取 借 阅 的 结束 日 期 
$flag1=isset($_POST[flag1])?$_POST[flag1]:"”"; // 获 取 按 指定 条 件 查询 的 复 选 框 值 
$flag2=isset($_POST[flag2])?$_POST[flag21:"; 1/ 获取 按 日 期 查询 的 复 选 框 值 
if($flag1=="a"){ // 如 果 按 指定 条 件 查询 ， 则 执行 以 下 语句 


$sql=mysqli_query($conn,"select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode, k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where $f like '%$key1%"); 


} 

if($flag2=="b"){ // 如 果 按 时 间 段 查询 ， 则 执行 以 下 语句 

S$sql=mysqli_query($conn,"select  b.borowTime,b.backTime,b.ifoack,r.barcode as readerbarcode,r.name, Kk.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
Where borrowTime between '$sdate' and '$edate"); 


} 
if($flag1=="a" &&. $flag2=="b"){ /| 如果 按 综合 条 件 查询 ， 则 执行 以 下 语句 
$sql=mysqli_query($conn,"select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name, 
k.id,k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=rid join tb_bookinfo k on 
b.bookid=k.id where borrowTime between '$sdate' and '$edate' and Sf like '%$key1%"); 
} 
} 
Sresult=mysqli_fetch_array($sql); /检索 查询 结果 
ifS$result==falsej{ /如果 查询 结果 不 存在 ， 则 弹出 提示 信息 
入 
<table width="100%" height="30" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td height="36" align="center"> 暂 无 图 书 借阅 信息 ! </td> 
</tr> 
</table> 
<?php 
} 
else{ // 否 则 ， 输 出 图 书 借阅 信息 
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?> 
<table width="946” border="1" cellpadding="6" cellspacing="0" bordercolor="#E1E1E1" style="border-collapse: 
collapse;"> 
<tr align="center" bgcolor="#DOE9F8"> 
<td width="13%"> 图 书 条 形 码 </td> 
<td width="27%"> 图 书 名称 </td> 
<td width="15%"> 读 者 条 形 码 </td> 
<td width="9%"> 读 者 名 称 </td> 
<td width="13%"> 借 阅 时 间 </td> 
<td width="13%"> 归 还 时 间 </td> 
<td width="10%"> 是 否 归还 </td> 
</tr> 
<?php 
dof 
if($result['ifback']=="0"){ /如果 “是 否 归 还 ”等 于 0， 则 输出 “未 归还 ” 
$ifbackstr=" 未 归还 "; 





} 
else{ // 和 如果“ 是 否 归 还 ”等 于 1， 则 输出 “已 归还 ” 


$ifbackstr=" 已 归还 "; 
} 
?> 
出 符合 查询 条 件 的 记录 "和 人 人 
<tr> 
<td><?php echo $result['barcode'];?></td> 
<td><a href="book_look.php?id=<?php echo $resultfid]; ?>"><?php echo $result [bookname']; ?></a></td> 
<td><?php echo $result['readerbarcode'];?></td> 
<td align="center"><?php echo $result[rname'];?></td> 
<td align="center"><?php echo $result[borrowTime'];?></td> 
<td align="center"><?php echo $result[backTime'];?></td> 
<td align="center"><?php echo Sifoackstr;?></td> 
</tr> 


ORO OCODRRRAA REORDER EAEOAn DEainiR Bho ia RhD ai Rohan halal Rolain eho lads toiainin eins iain iain ieiaie sieinieensini / 






<?php 
}while($result=mysqli_fetch_array($sql)); 
} 


?> 


6.9.7 单元 测试 


在 开发 完 图 书 借 还 模块 后 ， 对 该 模块 进行 了 单元 测试 。 当 操作 员 按 指定 字段 进行 条 件 查询 时 ， 能 
够 实现 图 书 借阅 查询 ， 当 操作 员 按 指定 的 时 间 段 进行 查询 时 ， 也 能 够 实现 图 书 借阅 查询 ;但 是 当 操作 
员 按 以 上 两 个 条 件 同 时 进行 查询 时 ， 则 不 能 实现 图 书 借阅 查询 。 

图 书 借 还 模块 的 表单 元 素 代 码 如 下 : 


<input name="flag" type="checkbox" class="noborder" value="a" checked> 
<input name="flag" type="checkbox" class="noborder" id="flag" value="b"> 








PHP 项 目 开发 全 程 实录 (第 4 版 ) 





完成 图 书 借阅 查询 的 SQL 语句 如 下 : 


if($flag: 

$sql=mysqli_query($conn,"select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode, k.bookname from tb_borrow b join tb_reader ron breaderid=rid join tb_bookinfo k on b.bookid=k.id where $f 
like '%$key1%"); 








$sql=mysqli_query($conn,"select b.borrowTime.,b.backTime,b.ifback,r.barcode as readerbarcode,r.name,k.id, 
k.barcode, k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where borrowTime between '$sdate' and '$edate"™); 


} 
if($flag=="a" && $flag=="b"){ 

$sql=mysqli_query($conn,"select b.borrowTime,b.backTime,b.ifoack,r.barcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where borrowTime between '$sdate' and '$edate' and $f like '"%$key1%"); 


} 


为 了 找 出 错误 的 原因 ， 笔 者 应 用 echo0 语 句 对 $flag 的 值 进行 输出 ， 输 出 的 结果 显示 $flag 等 于 b， 
不 执行 $flag 等 于 a。 这 充分 说 明 在 对 同名 复 选 框 进行 查询 时 ， 它 只 能 获取 最 后 一 个 复 选 框 的 值 。 

解决 该 问题 的 方法 是 将 两 个 复 选 框 分 别 定义 为 两 个 不 同 的 名 称 ,然后 赋予 不 同 的 值 ， 再 进行 判断 ， 
就 可 以 实现 按 指定 字段 、 指 定 的 条 件 进行 综合 条 件 查询 。 

因此 将 表单 中 的 HTML 标记 中 的 flag 进行 重 命名 ， 代 码 如 下 : 








<input name="flag1" type="checkbox" class="noborder" value="a" checked> 
<input name="flag2" type="checkbox" class="noborder" id="flag" value="b"> 


修改 后 的 完成 图 书 借阅 查询 的 SQL 语句 如 下 : 


if($flag1=="a"}{ 

$sql=mysqli_query($conn,"select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where $f like '%$key1%"); 
} 
if($flag2=="b"){ 

$sql=mysqli_query($conn,"select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where borrowTime between '$sdate' and '$edate"); 
} 
if($flag1=="a" && $flag2=="b"){ 

$sql=mysqli_query($conn,"select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
Where borrowTime between '$sdate' and '$edate' and $f like '%$key1%"); 
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6.10 开发 技巧 与 难点 分 析 


6.10.1 如 何 自动 计算 图 书 归 还 日 期 


在 图 书馆 管理 系统 中 会 遇 到 这 样 的 问题 : 在 借阅 图 书 时 ， 需 要 自动 计算 图 书 的 归还 日 期 。 
1. 图 书 归 还 日 期 


根据 图 书馆 还 书 的 规律 一 般 都 以 30 天 为 一 个 期 限 ， 因 此 在 图 书 归 还 时 ， 可 以 设置 一 个 固定 的 值 ， 
即 30 天 。 计 算 归 还 日 期 的 方法 如 下 : 

图 书 归还 日 期 = 系统 当前 日 期 + 借阅 天 数 固定 值 30 天 。 

自动 计算 图 书 归还 日 期 的 具体 代码 如 下 : 


date("Y-m-d",(time()+3600*24*30)) // 图 书 归还 日 期 








2. 续 借 图 书 归 还 日 期 


续 借 图 书 归还 日 期 是 在 原来 数据 库 保存 该 图 书 归还 日 期 〈 这 个 日 期 是 不 固定 的 ) 的 基础 上 再 次 借阅 
所 计算 的 时 间 ， 它 是 需要 根据 数据 表 中 保存 的 归还 日 期 来 计算 的 。 计 算 图 书 续 借 归还 日 期 的 方法 如 下 

续 借 图 书 归还 日 期 = 所 借 图 书 在 数据 表 中 的 归还 日 期 + 借阅 天 数 固定 值 30 天 。 

首先 应 用 substr0 函 数 分 别 取出 所 借 图 书 在 数据 表 中 原 定 的 归还 日 期 “月 ”“ 日 ”““ 年 ” 然后 应 用 
mktime0 函 数 计算 出 归还 日 期 的 时 间 戳 ， 最 后 应 用 date0 函 数 格式 化 日 期 为 “YYYYMM-DD” 格式。 
自动 计算 续 借 图 书 归还 日 期 的 代码 如 下 : 

Snew=9_GETIbackTimeT / 藤 取 传递 过 来 的 该 图 书 在 数据 表 中 的 归还 日 其 


// 更 新 续 借 期 ， 将 动态 获取 的 还 书 日 期 转化 为 时 间 戳 ， 然 后 再 求 出 续 借 后 的 还 书 日 期 
date("Y-m-d",(mktime(0, 0, 0, substr($new,5,2), substr($new,8,2), substr($new,0,4))+3600*24*30)); 








6.10.2 ”如 何 对 图 书 借阅 信息 进行 统计 排行 


在 图 书馆 管理 系统 的 首页 中 ， 提 供 了 显示 图 书 借阅 排行 榜 功 能 。 要 实现 该 功能 ， 最 重要 的 是 如 何 
获取 统计 排行 信息 , 这 可 以 通过 一 条 SQL 语句 实现 。 本 系统 中 实现 对 图 书 借阅 信息 进行 统计 排行 的 SQL 
语句 如 下 : 

select * from (select bookid,count(bookid) as degree from tb_borrow group by bookid) as borr join (select b.*, 

c.name as bookcasename,p.pubname,ttypename from tb_bookinfo b left join tb_bookcase c on b.bookcase= 


c.id join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as book on borr. 
bookid=book.id order by borr.degree desc limit 10 


下 面 将 对 该 SQL 语句 进行 分 析 : 
(1) 对 图 书 借阅 信息 表 进行 分 组 并 统计 每 本 图 书 的 借阅 次 数 ， 然 后 使 用 as 为 其 指定 别名 为 borr。 


® 
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代码 如 下 : 
(select bookid,count(bookid) as degree from tb_borrow group by bookid) as borr 
(2) 使 用 左 联接 查询 出 图 书 的 完整 信息 ， 然 后 使 用 as 为 其 指定 别名 为 book。 代 码 如 下 : 


(select b.*,c.name as bookcasename,p.pubname,ttypename from tb_bookinfo b left join tb_bookcase c on 
b.bookcase=c.id join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as book 


(3) 使 用 join on 语句 将 borr 和 book 连接 起 来 ， 再 对 其 按 统计 的 借阅 次 数 degree 进行 降序 排序 ， 
并 使 用 limit 子 句 限制 返回 的 行 数 。 


6.11 联接 语句 技术 专题 


在 实际 网 站 开发 过 程 中 ， 经 常 需要 从 多 个 表 中 查询 信息 ， 在 MySQL 数据 库 中 可 以 通过 连接 的 方 
式 实现 多 表 查 询 ， 连 接 方式 分 为 内 联接 和 外 联接 两 种 。 下 面 对 这 两 种 连接 方式 进行 详细 讲解 。 
6.11.1 ”内 联接 语句 


inner join 即 内 联接 查询 方式 ， 是 程序 开发 中 常用 的 连接 方式 。 内 联接 称 为 相等 联接 ， 它 返回 两 个 
表 中 的 所 有 列 ， 但 只 返回 在 联接 列 中 具有 相等 值 的 行 。 内 联接 查询 的 语法 格式 如 下 : 





select fieldlist 
from table1 [inner] join table2 
on table1.column=table2.column 





回 fieldlist， 要 查询 的 字段 列表 。 

回 tablel、table2: 为 要 连接 的 表 名 。 

回 inner: 可 选项 ， 表 示 表 之 间 的 连接 方式 为 内 联接 。 

回 ontablel.column=table2.column: 用 于 指明 表 tablel 和 表 table2 之 间 的 连接 条 件 。 

下 面 通过 内 联接 方式 实现 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 。 代 码 如 下 : 


$sql=mysqli_query($conn,"select 
tb_yg.userid,tb_yg.name,tb_yg.sex,tb_yg.age,tb_yg.tel,tb_yg.bm,tb_yg_info.gz from tb_yg inner join tb_yg_info 
on tb_yg.userid=tb_yg_info.ygid"); 

$info=mysqli_fetch_array($sql); 


6.11.2 ”外 联接 语句 


内 联接 返回 的 是 两 个 表 中 符合 条 件 的 数据 ， 而 外 联接 返回 部 分 或 全 部 匹配 行 ， 这 主要 取决 于 所 
建立 的 外 联接 的 类 型 。 外 联接 分 为 左 外 联接 和 右 外 联接 ， 下 面 对 这 两 个 外 联接 的 使 用 方法 进行 详细 
讲解 。 


局 
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1. 左 外 联接 (left outer join) 


左 外 联接 返回 的 查询 结果 包含 左 表 中 的 所 有 符合 查询 条 件 及 右 表 中 所 有 满足 连接 条 件 的 行 。 
MySQL 数据 库 中 使 用 左 外 联接 的 语法 格式 如 下 : 
select field 1[field2…] 
from table1 left [outer] join table2 
on join_condition 
[where search_condition] 


回 left outer join: 表示 表 之 间 通 过 左 外 联接 方式 相互 连接 ， 也 可 以 简写 成 left join。 

on join_condition: 指 多 表 建 立 连接 所 使 用 的 连接 条 件 。 

加 ”where search_condition: 可 选项 ， 用 于 设置 查询 条 件 。 

下 面 通过 左 外 联接 的 方式 建立 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 。 代 码 如 下 : 


$sql=mysqli_query($conn,"select * from tb_yg left outer join tb_yg_info on tb_yg.userid=tb_yg_infoygid "); 
$info=mysqli_fetch_array($sql); 
2. 右 外 联接 (right outer join) 


右 外 联接 返回 的 查询 结果 包含 左 表 中 的 所 有 符合 连接 条 件 以 及 右 表 中 所 有 满足 查询 条 件 的 行 。 
MySQL 数据 库 中 使 用 右 外 联接 的 语法 格式 如 下 : 


select field 1[field2…] from table1 right [outer] join table2 on join_condition [where search_condition] 








回 right outer join: 表示 表 之 间 通 过 右 外 联接 方式 相互 连接 ， 也 可 以 简写 成 right join。 

回 “outer: 可 选项 ， 表 示 表 之 间 的 联接 方式 为 完全 联接 。 

回 on join_condition: 指 多 表 建 立 联接 所 使 用 的 连接 条 件 。 

加 ”where search_condition: 可 选项 ， 用 于 设置 查询 条 件 。 

下 面 通过 右 外 联接 的 方式 建立 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 。 代 码 如 下 : 


$sql=mysqli_query($conn,"select * from tb_yg right outer join tb_yg_info on tb_yg.userid=tb_yg_info.ygid "); 
$info=mysqli_fetch_array($sql); 





6.12 本 章 总 结 


本 章 运 用 软件 工程 的 设计 思想 ， 通 过 一 个 完整 的 图 书馆 管理 系统 引导 读者 深入 了 解 系统 的 开发 流 
程 。 在 这 个 系统 的 实现 过 程 中 ， 除 了 应 用 一 些 基本 的 PHP 技术 之 外 ， 还 涉及 一 些 独特 的 技术 细节 ， 如 
权限 设置 、 多 表 查 询 技 术 等 需要 读者 掌握 ， 并 能 在 实际 的 操作 中 灵活 应 用 ， 举 一 反 三 。 
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博客 管理 系统 


(Apache+PHP+phpMyAdmin+MYySQL 5.5 实现 ) 


如 果 有 人 问 “ 什 么 是 \Web 2.0? ”， 息 必 没 几 个 人 能 说 得 清楚 ， 但 
是 要 问 什么 是 博客 ， 哪 怕 是 从 不 上 网 的 人 也 是 耳熟能详 的 了 。 博 容 正 
是 Web 2.0 概念 中 重要 的 组 成 部 分 之 一 《大 家 熟知 的 还 包括 IM 即时 
通 和 R99 阅读 器 )。 

Blog (博客 )， 全 名 Weblog， 后 来 缩写 为 Blog。Blogger 就 是 写 
Blog 的 人 人， 习惯 于 在 网 上 写 出 日 记 、 发 布 个 人 照片 、 展 示 个 性 自我 的 
用 户 群 体 。 对 于 Blog/Blogger 的 中 文 名 称 ， 有 翻译 成 “博客 ”， 也 有 
翻译 为 “网 志 ”， 但 大 多 数 人 都 已 经 认可 了 “博客 ”。 

通过 阅读 本 章 ， 可 以 学 习 到 : 


Lad 


博客 管理 系统 的 开发 流程 
进一步 掌握 如 何 做 项 目 需求 分 析 与 
系统 设计 

实现 一 个 简单 的 公告 栏 管理 模块 
掌握 不 同 的 图 片上 传 技 术 
掌握 一 种 采用 JavaScript+CSS 技术 来 实现 的 半 透 明 动 态 下 拉 菜 单 
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7.1 开发 背景 














| “视频 讲解 

博客 管理 系统 给 人 们 提供 抒发 个 人 情感 、 人 与 人 之 间 进 行 良好 沟通 的 平台 ， 博 客 拥 
有 真实 的 内 容 ， 可 以 通过 博客 记录 工作 、 学 习 、 生 活 和 娱乐 的 点 滴 ， 以 及 发 表 文 章 和 评论 ， 从 而 在 网 
上 建立 一 个 完全 属于 自己 的 个 人 天 地 ， 成 为 当今 网 络 最 为 个 性 化 和 平民 化 的 个 人 展示 空间 。 对 于 网 民 ， 
只 要 拥有 博客 ， 就 可 以 超越 现实 生活 ， 拥 有 不 同 以 往 的 全 新 网 上 生活 ， 对 于 博客 服务 商 ， 则 必须 从 功 
能 提供 转型 到 全 方位 社会 服务 的 提供 ， 建 立 虚拟 社会 ， 并 负责 维护 运行 ， 保 证 博客 日 常生 活 ， 对 于 社 
会 而 言 ， 有 利于 构建 和 谐 的 互联 网 空间 ， 维 护 和 谐 的 社会 环境 。 从 这 个 角度 来 说 ， 构 建新 生活 方式 
将 是 互联 网 发 展 的 一 个 里 程 碑 。 


7.2 需求 分 析 


信息 时 代 的 今天 ， 博 客 已 经 成 为 一 种 新 的 生活 方式 。 在 网 络 中 构建 一 个 赋 有 个 性 化 的 个 人 博客 ， 
提供 了 一 种 可 信任 的 和 实时 连通 的 网 络 环境 ， 通 过 网 络 开放 性 和 交互 性 的 特点 ， 让 用 户 在 任何 时 间 、 
任何 地 点 ， 通 过 网 络 方便 地 “生活 ”不仅 可 以 进行 信息 传递 与 获取 , 还 可 以 进行 群体 交流 和 资源 共享 ， 
展示 自我 ， 为 个 人 发 展 带 来 新 机 遇 。 
通过 对 多 个 博客 网 的 调查 分 析 ， 客 户 要 求 本 博客 管理 系统 具有 以 下 功能 
要 求 系统 采用 B/S 架构 ， 实 现 人 机 交互 。 
要 求 系统 界面 个 性 化 ， 色 彩 搭配 和 谐 ， 具 有 很 强 的 视觉 冲击 力 ， 操 作 简便 。 
要 求 突出 主题 ， 显 示 最 新 文章 和 公告 。 
要 求 游客 可 以 浏览 文章 、 浏 览 图 片 、 发 表 评论 。 
要 求 具有 强大 的 搜索 查询 功能 ， 实 现 精确 查询 和 模糊 查询 。 
完善 的 文章 管理 功能 ， 包 括 文章 的 发 表 、 删 除 ， 以 及 对 文章 的 评论 与 回复 。 
支持 图 片上 传 功 能 ， 可 以 上 传 各 种 类 型 的 图 片 。 
支持 好 友 功 能 。 
系统 运行 稳定 ， 安 全 可 靠 。 


加 加 加 网 网 加 网 网 加 


7.3 系统 设计 
7.3.1 系统 目标 


该 系统 主要 实现 如 下 目标 : 
加 ”系统 采用 B/S 架构 ， 实 现 人 机 交互 。 
回 “系统 界面 设计 以 浅 色 为 主 ， 美 观 友 好 ， 操 作 简 便 。 
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回 ”突出 重点 内 容 ， 显 示 最 新 文章 。 
非 登 录用 户 可 以 浏览 文章 、 浏 览 图 片 、 发 表 评论 。 
回 “全 面 的 搜索 查询 功能 ， 包 括 精确 查询 和 模糊 查询 。 
回 “完善 的 文章 管理 功能 ， 包 括 文章 的 发 表 、 删 除 ， 以 及 对 文章 的 评论 与 回复 。 
支持 图 片上 传 功能 。 
回 “支持 好 友 功能 。 和 
回 “支持 公告 栏 功能 。 了 
回 “系统 运行 稳定 ， 安 全 可 靠 。 曾 英 
a 
7.3.2 系统 功能 结构 {fi parla] [el fal fafa] fa! fa Fa fl [a Es 
览 | | 览 || 表 || 询 | | 传 | | 复 | | 传 | | 加 | | 除 |! | 户 | | 章 | | 论 | | 片 | | 告 
|| 文 | | 图 || 评 || 文 | | 文 | | 评 | | 图 | | 好 | | 好 || | 管 | | 管 | | 管 | | 管 | | 管 
博客 管理 系统 的 功能 结构 如 图 7.1 所 示 。 ialialialdlbs 片 | | 友 思 | | 可 | 加 | 到 加 杰 
















































































7.3.3 系统 功能 预览 图 7.1 博客 管理 系统 功能 结构 图 


为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 给 出 本 系统 的 几 个 页 面 运行 效果 图 ， 如 果 想 查 
看 完整 的 效果 图 ， 请 参见 资源 包 源 程序 。 

博客 管理 系统 的 首页 如 图 7.2 所 示 ， 该 页 面包 含 了 系统 大 部 分 的 功能 链接 ， 包 括 用 户 注册 、 用 户 
登录 、 文 章 浏览 等 。 用 户 注册 页 面 如 图 7.3 所 示 ， 该 页 面 显示 了 用 户 注册 时 需要 填写 的 资料 、 注 意 事 
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7.2 博客 首页 (资源 包 \TM\07\online\index.php) 图 7.3 用 户 注册 (资源 包 \TM\07\online\Register.php) 
浏览 文章 页 面 如 图 7.4 所 示 , 该 页 面 用 于 显示 文章 及 相关 的 评论 ， 也 可 以 发 表 评 论 。 发 表 文章 页 面 


如 图 7.5 所 示 ， 该 页 面 用 于 登录 用 户 发 表 文章 ， 包 括 文章 标题 、 文 字 编辑 区 和 文章 内 容 。 
图 片上 传 页 面 用 于 上 传 图 片 ， 如 图 7.6 所 示 ， 通过 该 页 面 用 户 可 以 将 图 片 或 照片 添加 到 博客 中 。 添 
加 好 友 页 面 如 图 7.7 所 示 ， 该 页 面 用 于 输入 用 户 好 友 的 详细 信息 ， 包 括 姓名 、 性 别 、 生 日 等 。 


[Ea 
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图 7.4 浏览 文章 (资源 包 \TM\07\online\article.php) 图 7.5 发 表 文章 (资源 包 \TM\07\online\file.php) 





ED mrrnn aol 














三 | | 二 pe 


图 7.6 图 片上 传 (资源 包 \TM\07online\add picphp) 图 7.7 添加 好 友 (资源 包 \TM\07\online\friend.php) 
7.3.4 系统 流程 图 


博客 管理 系统 的 流程 图 如 图 7.8 所 示 。 






































7.3.5 ”开发 环境 游客 ) 中 注册 | 冰 < 用 户 登 均 > (管理 员 
成 
在 开发 博客 管理 系统 平台 时 ， 该 项 目 使 用 的 软件 [ 症 | 荔 
名 普通 
开发 环境 如 下 。 让 
1. 服务 器 端 
操作 系统 : Windows 7。 查 国 济 文 | [图 | 隆 用 | [人 
服务 器 : Apache 2.4.18。 要 || 讲 | 澡 . 二 友 全 = 
PHP 软件 : PHP 7.0.12。 章 | | 论 | | 片 理 | | 理 | | 理 理 | | 理 
回 数据库: MySQL 5.5.47。 


图 7.8 博客 管理 系统 流程 图 
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MySQL 图 形 化 管理 软件 : phpMyAdmin-3.5.8。 
开发 工具 : PhpStorm 2016.3。 

浏览 器 : Google Chrome。 

分 辩 率 : 最 佳 效果 为 1680x1050 像素 。 


客户 端 


浏览 器 : 推荐 使 用 Google Chrome。 
分 辩 率 : 最 佳 效果 为 1680x1050 像素 。 


同 加 六 国 网 轿 回 





7.3.6 ”文件 夹 组 织 结构 Po 


用 于 存储 CSS 样式 表 文 件 


用 于 存储 数据 库 文件 


博客 系统 的 目录 比较 少 ， 结 构 比较 简单 ， 主 要 有 数 | 。 — 
据 库 链 接 文 件 目录 、CSS 样式 目录 、JS 脚本 目录 及 背景 一 
辐 片 目录 。 文 件 夹 组 织 结构 如 图 7.9 所 示 。 图 7.9 博客 管理 系统 文件 夹 组 织 结构 










7.4 ”数据 库 设计 





7.4.1 数据 库 分 析 


本 系统 属于 中 小 型 个 人 网 站 ， 毫 无 争议 的 ， 本 系统 采用 的 依然 是 PHP+MySQL 这 对 黄金 组 合 ， 
论 是 从 成 本 、 性 能 、 安 全 上 考虑 ， 还 是 从 易 操 作 性 上 考虑 ，MySQL 都 是 最 佳 选择 。 


7.4.2 ”数据 库 概念 设计 


通过 需求 分 析 和 功能 上 的 设计 ， 本 系统 规划 出 用 户 信息 实体 、 上 传 图 片 实体 、 朋 友 圈 实体 、 文 章 
实体 和 留言 实体 。 下 面 给 出 主要 的 实体 及 E-R 图 。 

用 户 信息 实体 包括 注册 用 户 的 详细 个 人 信息 ， 如 果 想 在 本 系统 中 进行 发 表 文 章 、 上 传 图 片 等 操作 ， 
则 必须 要 先进 行 注册 。 用 户 信息 实体 E-R 图 如 图 7.10 所 示 。 

上 传 图 片 实体 主要 包括 图 片 名 称 、 图 片 ia、 上 传 用 户 、 上 传 时 间 和 图 片 空间 等 , 实体 E-R 图 如 图 7.11 


























所 示 。 
上 传 用 户 
《用户 账 号 》 | 注册 用 户 用 户 性 别 a 
图 7.10 用 户 信息 实体 ER 图 图 7.11 上 传 图 片 实体 E-R 图 


@ 
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文章 实体 主要 包括 文章 id、 文 章 作者 、 文 章 标题 、 
文章 内 容 和 上 传 时 间 等 ， 实 体 ER 图 如 图 7.12 所 示 。 ETTY 


文章 作者 文章 内 容 
7.4.3 ”数据 库 物 理 结构 设计 


E 
根据 实体 E-R 图 和 本 系统 的 实际 情况 ,在 博客 管 


理 系统 的 数据 库 db_tmlog 中 需要 创建 6 张 数据 表 , 如 国 712. 入 融 实 从 ER 国 
表 7.1 所 示 。 

















表 7.1 db_tmlog 数据 库 中 的 数据 表 





表 类 型 整 理 说 了 明 
tb article MyISAM utf8 general ci 文章 列表 


tb_filecomment 评论 列表 
由 friend 好 友 列表 
tb public 公告 列表 
tb ps 图 片 列 表 
tb hser 用 户 列表 


下 面 来 具体 看 一 下 这 6 张 数 据 表 的 结构 设计 。 

1. tb_user (用户 列表 ) 

用 户 列表 主要 存储 用 户 的 个 人 信息 。tb_user 表 的 结构 如 表 7.2 所 示 。 
表 7.2 用 户 列表 结构 


名 字 | 类 型 | 整 理 | 空 | 默认 | 额 外 说 明 
id int(20) AUTO INCREMENT | ”自动 编号 
regname varchar(20) | utf8_general ci 用 户 名 
regrealname | varchar(20) | utf8 general ci 真实 姓名 




















regpwd varchar(40) | utf8_general ci 否 无 用 户 密码 
regbirthday date 否 0000-00-00 用 户 生 日 
regemail varchar(100) | utfs_general ci 否 无 E-mail 

regcity varchar(100) | utf8 general ci 否 无 所 在 城市 
Tegico varchar(50) | utf8 general ci 否 无 用 户头 像 
TegSeX varchar(4) utf8 general ci 否 无 用 户 性 别 


regqq varchar(40) | utf8 general ci 用 户 QQ 





reghomepage | varchar(100) | utf8 general ci 个 人 主页 
regsign varchar(200) | utf8 general ci 否 无 个 性 化 签名 
regintroduce mediumtext | utf8 general ci 自我 介绍 
让 varchar(20) | utfg_general ci | 否 | 无 | 登录 卫 


他 in) | lo | 是 否 管理 员 


301 
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2. tb_article (文章 列表 ) 
文章 列表 存储 的 是 用 户 发 表 过 的 文章 信息 。tb_article 表 的 结构 如 表 7.3 所 示 。 
表 7.3 文章 列表 结构 












各 字 说 明 




















id | intao) AUTO INCREMENT | “自动 编号 
content | mediumtext | ua general ci | 文章 内 容 
author | varchar(20) | us general ci | 文章 作者 
now | datetime 上 传 时 间 
title varchar(200) | utf8 general ci 文章 标题 











3. tb_filecomment (评论 列表 ) 


评论 列表 存储 的 是 用 户 对 文章 的 评论 ,包括 注册 用 户 和 游客 都 可 以 发 表 评论 。tb_filecomment 表 的 
结构 如 表 7.4 所 示 。 


表 7.4 评论 列表 结构 
名 字 | 类 型 | 整 理 | 空 | 默 认 | 额外 | 说 明 
id [mo | | 百 | 无 |AuromcREMENT| 自动 篇 
fd [nm | | 下 | | | x¥ia 
vsemame | vachar20 | us generalci| 否 | 无 | | 评价 用 
content 评价 内 容 


[tx | | 和 过 车 4” | 
E 0000-00-00 


4. tb_tpsc (图 片 列表 ) 


图 片 列表 存储 的 是 上 传 图 片 的 信息 ， 如 图 片 名 称 、 上 传 用 户 、 上 传 时间 等 。tb_tpsc 表 的 结构 如 表 7.5 
所 示 。 





表 7.5 图 片 列表 结构 










说 了 明 
AUTO INCREMENT | 自动 编号 














tpmec Varchar(30) | utf8 general ci | 图 片 名 称 
file blob | ”三 进 制图 片 
author | ”上 传 用 户 





Scsj 


上 传 时 间 





5. tb_friend (好 友 列表 ) 
好 友 列 表 主 要 记录 了 姓名 、 性 别 、 生 日 等 好 友 的 个 人 信息 。tb_friend 表 的 结构 如 表 7.6 所 示 。 


@ 
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表 7.6 好 友 列表 结构 









































名 字 类 型 整 理 空 额 外 说 了 明 
id int(4) 否 AUTO INCREMENT 自动 编号 
name varchar(50) | utf8 general ci 否 好 友 姓 名 
SeX varchar(10) | utf8 general ci 否 好 友 性 别 
bir date 否 好 友 生 日 
city varchar(50) | utf8 general ci 否 所 在 城市 
address varchar(100) | utf8 general ci 否 家 庭 住址 
postcode varchar(6) utf8_general ci 否 邮政 编码 
email varchar(50) | utf8 general ci 否 E-mail 

tel varchar(20) | utf8 general ci 否 家 庭 电话 
handset varchar(20) | utf8 general ci 否 手机 号 码 
QQ varchar(20) | utf8 general ci 否 好 友 QQ 
Username | varchar(20) | utf8 general ci | 否 | 无 | 用 户 名 





6. tb_public (公告 列表 ) 
公告 列表 主要 记录 了 网 站 情况 、 博 客 系统 的 版 本 情况 或 是 网 站 活动 等 。 公 告 列表 的 结构 如 表 7.7 


表 7.7 公告 列表 结构 
名 字 | 类 型 | 整理 | 空 | 默 认 | 额外 | 说 明 
id [am | | 不 | 无 |auro INCREMENT | 自动 编号 
tde | vachar50) |ufsgenerldi| 否 | 无 | | 公告 记 
content | vachar2o |ufs generalci| 否 | 无 | | 公告 容 
potime [de | | | 无 | | 发 Hi 





7.5 首页 设计 





本 系统 首页 页 面 设计 简洁 ， 主 要 包括 以 下 3 部 分 内 容 。 

回 ”首部 导航 栏 : 包括 首页 链接 、 注 册 和 登录 模块 。 

回 ” 左 侧 显 示 区 : 包括 最 新 文章 、 最 新 图 片 和 系统 时 间 模 块 。 游 客 主要 通过 该 区 域 浏览 文章 、 浏 
览 图 片 及 发 表 评 论 。 

回 ” 主 显示 区 : 为 系统 公告 栏 ， 显 示 系统 及 网 站 的 最 新 咨询 。 

本 案例 中 提供 的 首页 如 图 7.13 所 示 。 该 首页 在 本 书 资源 包 中 的 路 径 为 \TM\07\tmlog\index.php。 
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-博信 六 本 六 儿 动 和 局 一 
一 人 天 作 全 的 由 可 专属 志 一 

-- 王 划 1 有 上 线 ! 6CCD 情 吾 过 福 这 乏 玩 一 
一重 障 ! BCCD 傅 大 年度 尼 过 活动 各 竺 上 线 一 
一 FEBCCD 合 襄 风 重 新 开放 一 











图 7.13 博客 管理 系统 首页 


7.5.2 首页 技术 分 析 


在 首页 主 显示 区 ， 是 一 个 公告 栏 模块 。 公 告 栏 主要 用 于 公布 系统 版 本 的 更 新 或 升级 情况 、 网 站 的 
最 新 活动 安排 等 ， 也 可 以 链接 一 些 用 户 的 精彩 文章 。 本 系统 的 公告 栏 模块 是 通过 <marquee> 标 签 来 实现 
的 。<marquee> 标 签 是 HTML 自 带 的 ， 也 是 初学 者 最 常用 的 公告 栏 实现 方式 。 使 用 <marquee> 标 签 ， 可 
以 实现 文字 或 图 片 的 滚动 效果 ， 增 加 了 实用 性 的 同时 ， 也 增加 了 观赏 性 。 下 面 就 来 学 习 这 个 标签 的 常 
用 方法 及 属性 。 


入 站 <marquee> 标 答 是 微软 与 网 景 等 公司 私 制 的 ， 从 来 没有 被 W3C 当 作 正式 标签 来 使 用 ， 所 
以 除了 下 浏览 器 外 ， 有 些 属性 是 不 被 其 他 浏览 器 所 支持 的 ， 在 使 用 时 一 定 要 留意 。 


1. <marquee> 标 签 的 文字 移动 属性 


<marquee> 标 签 的 特点 就 是 可 以 使 文字 或 图 片 动 起 来 ， 在 早 些 时 候 ， 这 可 是 一 个 了 不 起 的 技术 。 随 
着 W3C 标准 的 逐渐 完善 ，<marquee> 标 签 已 经 越 来 越 少 有 用 武之 地 了 ， 但 有 些 技术 是 永远 不 变 的， 如 
<marquee> 标 签 中 的 属性 名 称 和 属性 值 , 了 解 CSS 样式 表 和 JavaScript 脚本 语言 的 人 会 感觉 到 非常 熟悉 ， 
因为 这 些 属性 名 称 和 属性 值 都 经 常 被 提 及 和 使 用 。<marquee> 标 签 用 的 文字 移动 属性 及 说 明 如 表 7.8 所 示 。 


表 7.8 <marquee> 标 签 常 用 的 文字 移动 属性 及 说 阴 








属性 说 明 应 用 举例 
| 标签 基本 语法 , 除了 文字 外 , 还 可 以 是 
marquee 无 
图 片 
文字 移动 属性 , 分 别 表示 从 右 往 左 、 从 | <marquee direction="up"> 从 下 到 上 移动 
左 往 右 、 从 下 到 上 、 从 上 到 下 <marquee> 





<marquee> 你 好 ，PHP</marquee> 
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属性 名 称 属 性 值 属性 说 明 应 用 举例 
文字 移动 方式 , 分 别 表示 沿 同一 方向 不 
Behavior scroll,slide.alternate | 停 滚动 、 只 滚动 一 次 、 在 两 个 边界 内 来 
回 滚动 





<marquee behavior="scroll"> 不 停 的 循环 
播放 </marquee> 





<marquee loop=5 behavior=slide> 只 循环 
滚动 3 次 </marquee> 

<marquee scrollmount=25> 我 漂 起 来 了 !! 
</marquee> 

<marquee scrolldelay="500" scrollamount= 

100> 停 停 走 走 </marquee> 


loop 数值 1.2.3… 循环 次 数 ， 不 指定 则 表示 为 无 限 循环 


scrollmount | 数值 10.20.30… 滚动 速度 ， 数 值 越 大 ， 速 度 越 快 





Scrolldelay | 数值 100.200.300… | 延 时 ， 每 动 一 次 ， 停 止 的 时 间 








2. <marquee> 标 签 的 外 观 设置 


<marquee> 标 签 不 仅 可 以 设置 文字 移动 效果 , 还 能 设置 标签 的 外 观 。 外 观 包 括 对 齐 方式 、 标 签 底 色 、 
面积 等 。<marquee> 标 签 常 用 的 外 观 属 性 及 说 明 如 表 7.9 所 示 。 


表 7.9 <marquee> 标 签 常用 的 外 观 属性 及 说 明 

应 用 举例 

<marquee align="middle"> 我 在 中 间 </marquee> 
<marquee bgcolor="#DEEBEF"> 看 到 了 吗 ? 
</marquee> 


滚动 栏 高 度 <marquee height="15"> 不 要 ， 太 矮 了 </marquee> 
滚动 栏 宽度 <marquee width="100"> 哇 ! 好 大 的 房子 啊 </marquee> 





3. <marquee> 标 签 的 其 他 属性 


除了 以 上 的 常规 设置 ，<marquee> 标 签 还 有 两 个 特殊 的 属性 ， 即 start0 和 stop0， 它 们 可 以 配合 
JavaScript 事件 产生 很 有 意思 的 效果 。 例 如 : 


onMouseOut="this.start()"; // 当 鼠标 移出 该 区 域 时 ， 开 始 滚动 
onMouseOver="this.stop()"; // 当 鼠标 进入 该 区 域 时 ， 停 止 滚动 





7.5.3 ”首页 的 实现 过 程 


博客 管理 系统 采用 二 分 栏 结构 ， 表 单 布 局 。 具 体 实 现代 码 如 下 : 
倒 程 01 代码 位 置 : 资源 包 \TM\07\tmlog\index.php 





<?php 

session_start(); /开启 session 支持 
include "Conn/conn.php"; // 包 含 数据 库 链接 文件 
include "function.php"; /包含 函数 文件 
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?> 
<table width="920" border="0" align="center" cellpadding="0" cellspacing="0"> 

<tr align="right" valign= "top"> 
<td height="149" colspan="2" background="images/head.jpg"> 








<!-- 上 部 导航 栏 --> 
</td> 
</tr> 
<tr> 
<td width="236" height="501" style="background:url(images/left.jpg) no-repeat"> 
sm < 上 -- 左 侧 导航 栏 -> 
</td> 
<td width="684" height="501" align="center" background=" images/right.jpg"> 
<!- 系统 公告 区 一 一 一- > 
<?php 
$p_sql = "select * from tb_public order by id desc"， // 从 数据 表 中 读 取 数据 
$p_rst = mysqli_query($link,$p_sql); /| 执行 SQL 语句 
eg 
< 一 一 一 一 一 一 一 一 -一 一 一 一 -<marqUuee> 标 签 开始 ---------------------------------------- 一 > 
© <marquee onMouseOver=this.stop() onMouseOut=this. start() scrollamount=2 scrolldelay= 
7 direction=up> 
<?php 
while($p_row = mysqli_fetch_row($p_rst)X{ // 循 环 输出 公告 标题 
gs 
@ <a href="#" style="height:30px; line-height:30px; font-size:14px” onclick="wopen=open 


(show_pub.php?id=<?php echo 
$p_row[0]; ?>",",'height=200,width=500,scollbars=no')">--<?php echo $p_row[1]; ?>--</a><br> 





<?}?> 
</marquee> 
< > 
</td> 
</tr> 
</table> 
Ah 代码 贴 二 


@ <marquee> 标 签 : <marquee> 标 签 中 包含 了 start 和 stop 两 个 特殊 的 属性 ， 还 有 速度 属性 scollamount 和 延迟 属性 


scrolldelay， 而 公告 栏 的 高 度 ( height ) 和 宽度 ( width ) 则 是 通过 CSS 样式 设置 的 。 


@ <?php echo $p_ row[1]; ?>: 循环 输出 公告 标题 ， 并 为 每 个 公告 添加 超 链接 ， 这 里 用 到 了 JavaScript 脚本 的 open() 


方法 ， 该 方法 的 作用 是 打开 一 个 新 窗口 ， 同 时 可 以 对 窗口 样式 做 出 设置 。open( 方 法 的 一 般 格式 为 : open(" 链 接 的 Url"," 


自 定义 名 称 "," 窗 口 样式 ")。 











7.6 文章 管理 模块 设计 








7.6.1 文章 管理 模块 概述 
对 一 个 博客 系统 来 说 ， 文 章 管理 是 最 基本 的 功能 ， 但 同时 也 是 最 复杂 的 一 个 功能 。 本 系统 的 文章 





第 7 章 博客 管理 系统 ( Apache+PHP+phpMyAdmin+MySQL 5.5 实现 ) 


管理 模块 包括 “添加 博客 文章 “查找 博客 文章 ”“ 管 理 我 的 博客 “发 表 评 论 ” “删除 文章 ”和 “人 删 
除 评论 ”等 6 大 功能 。 其 中 ， 普 通用 户 只 能 删除 自己 的 文章 及 对 文章 的 评论 ， 只 有 管理 员 才 有 权 删 除 
任何 一 篇 文章 及 回复 。 文 章 管理 模块 的 框架 如 图 7.14 所 示 。 





Fi ms 
入 、N 
/ \ 
/ | 
\ / 
\ / 
* J 
As i 
普通 用 户 超级 用 户 


图 7.14 文章 管理 模块 框架 图 
7.6.2 文章 管理 模块 技术 分 析 


想 要 使 用 文章 管理 模块 ， 前 提 是 用 户 必 须 登录 ， 匿 名 用 户 是 无 法 访问 这 些 功 能 的 ， 想 要 删除 文章 
和 评论 ， 前 提 是 当前 用 户 要 么 是 管理 员 权 限 ， 要 么 是 文章 拥有 者 ， 否 则 不 会 显示 删除 功能 的 。 这 两 方 
面 的 控制 都 需要 session 的 配合 ， 本 节 就 来 讲 一 下 session 的 应 用 及 常见 的 问题 处 理 。 

session 的 中 文 译名 为 “会 话 ”， 是 指 用 户 从 进入 网 站 开始 ， 直 到 关闭 网 站 这 段 时 间 内 ,所 有 网 页 共 
同 使 用 的 公共 变量 的 存储 机 制 。 session 比 cookie 更 有 优势 : session 是 存储 在 服务 器 端的 , 不 易 被 伪造 ; 
session 的 存储 没有 长 度 限制 ，session 的 控制 更 容易 等 。 

PHP 中 的 session 功能 一 直 令 许多 初学 者 望 而 生 戎 ， 因 为 经 常 出 现 一 些 莫 名 其 妙 的 错误 ,而 又 不 知 
道 如 何 去 解 决 。 其 实 ， 大 多 数 的 错误 都 是 因为 对 session 的 配置 不 了 解 而 使 用 方法 不 正确 造成 的 ， 在 
php.ini 中 对 session 的 配置 如 表 7.10 所 示 。 


表 7.10 session 的 常用 配置 选项 














配置 选项 说 阴 
session.save_path = c:/temp | 保存 session 变量 的 目录 ， 在 Linux/UNIX 下 为 /mp 
session.ues cookies=1 | 是 否 使 用 cookie 
session.name = PHPSESSID | 表示 会 话 人 D 
session.auto_start =0 | 是 否 自动 启用 session， 当 为 1 时 ， 在 每 页 中 就 不 必 调 用 session_start0 函 数 了 





session.cookie lifetime =0 | 设 定 cookie 送 到 浏览 器 后 的 保存 时 间 ， 单 位 为 秒 。 默 认 值 为 0， 表示 直到 浏览 器 关闭 
session.cookie path=/ cookie 有 效 路 径 
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配置 选项 


说 阴 





session.cookie domain = 


有 效 域名 





session.serialize handler = php 


定义 序列 化 数据 的 标识 ， 本 功能 只 有 WDDX 模块 或 PHP 内 部 使 用 ， 默 认 值 为 PHP 





session.gc probability = 1 


设 定 每 次 临时 文件 开始 处 理 的 处 理 概率 。 默 认 值 为 1 





session.gc_ maxlifetime = 1440 


设 定 保存 session 的 临时 文件 被 清除 前 的 存活 秒 数 





session .referer check= 


决定 参照 到 客户 端的 session 代码 是 否 要 删除 。 有 时 出 于 安全 或 其 他 考虑 ， 会 设 定 不 
删除 。 默 认 值 为 0 





Session.cache limiter = nocache 


设 定 session 缓冲 限制 





session.cache expire = 180 


文档 有 效 期 ， 单 位 为 分 钟 





session.save_ handler = files 


用 于 保存 session 变量 ， 默 认 情 况 下 用 文件 





对 于 初学 者 来 说 ，session 在 php.ini 中 不 需要 特意 去 改动 ， 因 为 安装 时 会 根据 操作 系统 自行 做 出 适 

当 的 调整 。 只 在 少数 的 几 项 ， 如 session 存活 周期 (session.cookie lifetime = 0)、 自 动 开 启 session 
(session.auto_start) 等 稍 加 改动 即 可 。 

PHP 主要 是 通过 会 话 (session ) 处 理 函 数 来 对 session 进行 控制 和 使 用 的 。 常 用 的 处 理 函数 如 表 7.11 


所 示 。 


函数 
Session start():; 
$_ SESSION[mname'] = value: 
session idO 


isset($_SESSION[mame]) 


Session_Iegenerate id() 
session_name() 
unset($_SESSION['name']) 


表 7.11 PHP 常用 的 会 话 处 理 函 数 
函数 说 明 

开启 session 或 返回 已 经 存在 的 session 
注册 一 个 session 变量 
设 定 或 取得 当前 的 session id 值 
检测 指定 的 session 值 是 否 存在 。isset 不 只 可 以 检测 session， 还 可 以 检测 其 他 类 型 ， 如 
isset($_ POST[name'])、isset($_GET[mame]) 等 
更 改 session id 的 值 
返回 或 改变 当前 session 的 name 
删除 名 为 name 的 session 





session_destroyO 


结束 当前 会 话 ， 删 除 所 有 session 


全。 注意 (1) 如 果 要 改变 当前 session 的 name 值 ， 必 须 在 session() 之 前 调用 session nameO 函 数 ， 
而 且 session_name 不 能 全 部 是 数字 ， 否 则 会 不 停 地 生成 新 的 Session id。 
(2) 不 可 以 写成 unset($_SESSION)， 这 样 会 禁止 整个 会 话 的 功能 。 


在 本 节 最 后 ， 介 绍 使 用 session 时 要 注意 的 几 个 问题 。 
1. 尽 可 能 地 将 session_start() 放 到 第 1 行 
这 种 情况 是 新 手 最 容易 犯 的 错误 。 产 生 的 错误 代码 为 : 


Warning: session_start() [function.session-start]: Cannot send session cache limiter - headers already sent… 


@ 
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其 原因 就 是 在 使 用 session_start0 之 前 ， 就 有 HTML 代码 输出 了 。 也 许 有 的 读者 会 说 :“ 没 有 ， 
session_start0 之 前 没有 任何 代码 ， 绝 对 没有 。” 那 么 ,请 检查 你 的 程序 是 不 是 有 空 行 ， 或 类 似 echo 语句 
的 输出 。 如 果 有 ， 请 去 掉 ， 因 为 就 算是 一 个 小 小 的 空格 都 是 不 可 以 的 。 所 以 ， 为 了 避免 这 类 错误 的 发 
生 ， 尽 可 能 地 将 session_start0 放 到 第 1 行 。 


2. 在 使 用 session 之 前 一 定 要 先 写 session_start() 


大 多 数 读者 在 使 用 session 之 前 都 能 先 调 用 session_start0 函 数 , 但 对 于 session destroy0O 函 数 却 经 常 
忽略 。session_destroy0 虽 然 是 结束 当前 会 话 并 删除 所 有 session， 但 在 删除 之 前 ， 也 要 先 开启 session 支 
持 才 可 以 ， 不 然 会 产生 这 样 的 错误 代码 : 


session_destroy() [function.session-destroy]: Trying to destroy uninitialized session in… 





所 以 ， 凡 是 在 使 用 session 或 session 函数 的 页 面 中 ， 都 要 加 上 session_start0 这 人 句 话 。 

3. 删除 所 有 session 

如 果 想 删除 所 有 session, 但 又 不 想 结束 当前 会 话 ， 用 unset 一 个 一 个 删除 实在 是 太 麻烦 了 ， 最 简单 
的 办 法 就 是 将 一 个 空 数组 赋 给 $_SESSION， 如 $_SESSION = array0， 这 样 就 解决 了 。 
7.6.3 添加 文章 的 实现 过 程 


国 添加 文章 模块 使 用 的 数据 表 : tb_article 
当 用 户 登录 后 ， 系 统 会 直接 进入 到 文章 添加 页 (file.php)， 也 可 以 通过 单 击 “ 文 章 管理 ”/“ 添 加 
博客 文章 ” 回 到 file.php 页 。 添 加 文章 页 面 的 运行 结果 如 图 7.15 所 示 。 








ET ] 





月 有 训 ， 到 请 -人 
wm: |Z) a Fs 5 3 He We 
Ew 和 一 些 4 和 本 全 人。 竺 名 大 了 反而 看 二 江东 了 ， 教 好 HTF 


生息 这 和 半生 个 
It] 





全 本 个 近 失 0 交 的， 


E23E 
图 7.15 添加 文章 页 面 的 运行 结果 
添加 文章 页 为 一 个 发 布 表 单 ,包括 文章 主题 、 文 字 编 辑 、 文章 内 容 等 元 素 。 部 分 表单 元 素 如 表 7.12 


所 示 。 
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表 7.12 添加 文章 页 面 的 主要 表单 元 素 


























名 称 | 元 素 类 型 重要 属性 含 义 
form method="post" action="check file.php" 添加 文章 表单 
text id="txt title" size="68" 文章 标题 
ei name="font" ER Wo 文章 字体 

‘onChange="showfont(this.options[this.selectedIndex].value)" 
lass="wenbenkuang" 
select RE - 字体 大 小 
onChange="showsize(this.options[this.selectedIndex].value)" 
dele wohanpes "showcolor hs optons this seleot Inds value)" 字体 颜色 
name="color" Size="1" class="wenbenkuang" id="select" 
textarea cols="75" rows="20" id="file" style="border:0px:width:520px:" 文章 内 容 
submit id="btn_tj" value=" 提 交 " onClick="return checkO:" “提交 ”按钮 


当 用 户 填写 完 博客 主题 和 文章 内 容 后 ， 单 击 “ 提 交 ” 按 钮 ， 系 统 将 跳 转 到 处 理 页 (check_file.php) 
进行 处 理 。 在 处 理 页 中 ， 将 传 过 来 的 文章 标题 、 文 章 作者 和 文章 内 容 等 参数 组 成 insert 语句 ， 并 最 终 保 
存 到 数据 表 中 。 如 果 添 加 信息 成 功 ， 系 统 返回 到 本 页 ， 可 继续 执行 添加 操作 ;如 果 添 加 失败 ， 则 返回 
到 上 一 步 。 程 序 的 关键 代码 如 下 : 


倒 程 02 ”代码 位 置 ， 资源 包 \TM\07\tmlog\check file.php 





<?php 

header("Content-type: text/html; charset=utf-8"); /设置 文件 编码 格式 
session_start(); /开启 session 支持 

include "Conn/conn.php"; // 包 含 数据 库 链 接 文件 
date_default_timezone_set("PRC"); /设置 时 区 
if(isset($_POST['btn_1])X // 判 断 传 值 页 面 
S$title=$_POST[txt_title]; // 取 得 文章 标题 
$author=$_SESSION[username]; // 取 得 作者 
$content=$_POST[file]; /取得 文章 内 容 
$now=date("Y-m-d H:i:s"); /使 用 date() 函 数 生 成 发 布 时 间 


/生成 insert 语句 */ 
$sql="Insert Into tb_article (title,content,author,now) Values ('$title','$content','$author','$now’)"; 
$result=mysqli_query($link,$sql); /执行 insert 语句 
/根据 $result， 返 回 结果 当 
if($resultY{ 
echo "<script>alert(' 恭 喜 您 ， 你 的 文章 发 表 成 功 山 );window.location.href='file.php';</script>"; 


else{ 

echo "<script>alert(' 对 不 起 ， 添 加 操作 失败 山 );history. go(-1);</script> "; 
} 
} 


?> 





人 
说 明 文章 添加 页 面 中 ， 使 用 了 部 分 UBB 语法 ， 由 于 UBB 语法 不 属于 本 书 的 范 时， 所 以 这 里 
不 做 讲解 ， 请 感 兴趣 的 朋友 查看 相关 的 书籍 。 
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7.6.4 文章 列表 的 实现 过 程 


国 。 查看 文章 列表 使 用 的 数据 表 : tb_article 
单 击 “ 文 章 管理 ”/“ 我 的 文章 ”， 将 显示 用 户 发 表 过 的 文章 列表 。 文 章 列表 页 面 (myfiles.php) 的 
运行 结果 如 图 7.16 所 示 。 








| 


、 钨 国 多 表 的 图形 国信 处 理 技术 

百 实 府 方 -关于 字符 刷 纹 操作 

北纬 不 振 移 一 对 握 库 编 得 技术 

PHF 间 缚 枉 贞 区 并 本 不 重要 

涛 PH 的 运 贡生 

PMP 型 

、 世 界 各 复 东 ， 也 界 根 简 兰 
珍惜 每 一 个 现在 

TR: 1/3 页 记录: 8 条 


图 7.16 文章 列表 页 面 的 运行 结果 
文章 列表 页 面 使 用 了 分 页 技术 和 do…while 循环 语句 来 输出 文章 标题 。 程 序 关键 代码 如 下 : 


倒 程 03 ”代码 位 置 ， 资源 包 \TM\07\tmlog\myfiles.php 
<?php 
session_start(); /开启 session 支持 
include "Conn/conn.php"; // 包 含 数 据 库 链 接 文件 
include "check_login.php"; // 包 含 权限 检查 文件 
?> 
/分 页 4 
<?php 
/ 风 page 为 当前 页 ， 如 果 $_GET["'page'] 不 存在 ， 则 初始 化 为 1*/ 

if (lisset($_GET[page]))f 


$page=1; 
jelsef 

$page=$_GET[page]; 
} 
Spage_size=10; /每 页 显示 10 条 记录 
$query="select count(*) as total from tb_article where author = ".$_SESSION['username']." order by id 

desc"; 

Sresult=mysqli_query(S$link, $query); /查询 符合 条 件 的 记录 总 条 数 
$data = mysqli_fetch_array($result); /将 查询 结果 返回 到 数组 
$message_count=$data[total]; /获取 查询 总 记录 数 
$page_count=ceil($message_count/$page_size); 。 // 根 据 记录 总 数 除 以 每 页 显示 的 记录 数 求 出 所 分 的 页 数 
Soffset=($page-1)*$page_size; // 计 算 下 一 页 从 第 几 条 数据 开始 循环 
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$sql=mysqli_query($link,"select id,title from tb_article where author = ".$_SESSION['username']." order by 
id desc limit $offset, $page_size"); 

Sinfo=mysqli_fetch_array($sql); 
?> 


<?php 
1* 输 出 结果 集 */ 
if(!$infoX{ 
?> 
<tr><td align="center"><font color=#ff0000> 您 还 未 添加 任何 文章 !</font></td></tr> 
<?php 
Jelse{ 
$i=$offset+1;// 文 章 序号 
@ do /do…while 循环 开始 
?> 
<tr> 
<!- 显 示 文章 序号 和 文章 标题 -> 
<td height="35" style="padding-left:30px"><a style="font-size:14px; color:#0066FF" href="showmy.php?file_id= 
<?php echo $info['id];?>"><?php echo $i."、".$info['title];?></a> </td> 
</tr> 
<?php 
$i=$i+1; 
}while($info=mysqli_fetch_array($sql)); /| 循环 结束 
?> 
</table></td> 
</tr> 
</table></td> 
</tr> 


<!- 翻 页 条 --> 

<td width="33%">&nbsp;&nbsp; 页 次 : <?php echo $page;?>/<?php echo $page_count;?> 页 &nbsp; 记 录 : <?php 
echo $message_count;?> 条 &nbsp; </td> 

<td width="67%" align="right" class="hongse01"> 

<?php 

人 * 如 果 当前 页 不 是 首页 */ 

if($page!=1){ 

/* 显 示 “ 首 页 ” 超 链接 

echo "<a href=myfiles.php?page=1> 首 页 </a>&nbsp;"; 

/* 显 示 “ 上 一 页 ” 超 链接 

echo "<a href=myfiles.php?page=".($page-1)."> 上 一 页 </a>&nbsp; 


} 

让 如 果 当 前 页 不 是 尾 页 */ 

if($page<$page_count) 

{ 

/* 显 示 “ 下 一 页 ” 超 链接 六 

echo "<a href=myfiles.php?page=".($page+1)."> 下 一 页 </a>&nbsp;"; 
* 显 示 “ 尾 页 ” 超 链接 */ 

echo "<a href=myfiles.php?page=".$page_count."> 尾 页 </a>"; 

办 


?> 
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二 代码 贴 十 

@ do…while 循环 和 while 循环 : do…while 循环 是 先 执行 {} 中 的 代码 段 ， 然 后 判断 while 中 的 条 件 表达 式 是 否 成 立 ， 
如 果 成 立 返 回 true， 重 复 输出 {} 中 的 内 容 ， 否 则 结束 循环 ， 执 行 while 下 面 的 语句 。while 循环 是 先 判 断 while 中 的 表达 
式 ， 当 返回 true 时 ， 再 执行 {} 中 的 代码 。 两 者 的 区 别 是 : do…while 循环 比 while 循环 多 输出 1 次 结果 。 


7.6.5 查看 文章 、 评 论 的 实现 过 程 


国 查看 文章 、 评 论 使 用 的 数据 表 : tb_ article、tb_filecomment 
单 击 列表 中 任意 一 个 文章 标题 , 都 会 看 到 对 应 的 文章 内 容 和 文章 评论 。 查 看 文章 页 面 (showmyphp) 
的 运行 结果 如 图 7.17 所 示 。 








ION 104 WA mc R2017-12-13 16:47:34 
这。 个 当 在 ,不 入 宙 2 万 相配， 天 坟 全 。 。。 ZX 雪 区 


IR: 1/ 页 己 果 : 1 莱 
Re 


琶 ] 本 
图 7.17 查看 文章 页 面 的 运行 结果 


系统 根据 当前 页 面 传 过 来 的 文章 id 值 从 数据 表 tb_article 中 返回 对 应 的 文章 信息 (包括 文章 id、 文 
章 作 者 、 文 章 标题 、 文 章 内 容 和 发 表 时 间 )、 输 出 文章 信息 后 ， 开 始 查找 表 tb_filecomment 中 fileid 字 
段 值 等 于 文章 id 的 所 有 评论 集 ， 并 通过 分 页 显示 出 来 。 显示 文章 页 面 (showmy.php) 的 关键 代码 如 下 : 


倒 程 04 代码 位 置 ， 资源 包 \TM\07\tmlog\showmy.php 


<?php 

session_start(); /开启 session 支持 

include "Conn/conn.php"; /包含 数据 库 链接 文件 

include "check_login.php"; /包含 权限 检查 文件 
S$file_id=$_GETT'file_id']; // 取 得 文章 ID 号 

$bool = false; // 判 断 用 户 删 除权 限 , 默认 的 false 没有 权限 


?> 


< 上 -首部 导航 栏 -> 
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<! -- -- 一 





<!-- 文 章 显示 区 --> 
<?php 
S$sql=mysqli_query($link,"select * from tb_article where id = ".$file_id);”// 根 据 文章 ID 号 查找 文章 
Sresult=mysqli_fetch_array($sql); // 返 回 结果 集 
?> 
Se 显示 文章 J 





<table width="666" border="0" cellpadding="5" cellspacing="1" bordercolor="#D6E7AS" bgcolor="#FFFFFF" 
class="i_table"> 
<tr bgcolor="#FFFFFF" height="30"> 

<td width="14%" align="center"> 博 客 ID 号 </td> 

过 = 显示 文章 id a 

<td width="15%"><?php echo $result['id]; ?></td> 

<td width="11%" align="center"> 作 者 </td> 

< 一 一 一 一 一 一 显 示 文章 作者 一 一 一 一 > 

<td width="18%"><?php echo $result['author]; ?></td> 

<td width="12%" align="center"> 发 表 时 间 </td> 

< 一 一 一 一 一 一 一 显示 发 表 时 间 -一 一 一 一 一 一 一 -一 一 

<td width="20%"><?php echo $result['now']; ?></td> 

</tr> <tr bgcolor="#FFFFFF" height="30"> 

<td align="center"> 博 客 主题 </td> 

< 一 一 -一 -一 -一 一 -一 -文章 主题 

<td colspan="5">&nbsp;&nbsp;&nbsp;&nbsp;<?php echo $result['title’]; ?></td> 

</tr> <tr bgcolor="#FFFFFF"> 
<td align="center"> 文 章 内 容 </td> 











> 


a 一 文章 内 容 -ee ; > 
<td colspan="4" style="line-height:1.5">&nbsp;&nbsp;&nbsp;&nbsp;<?php echo $result['content]; ?></td> 
<td> 
<?php 
/* 判 断 登录 用 户 是 否 为 管理 员 ， 或 者 是 文章 的 作者 */ 
if($_SESSION['fig]==1 or ($_SESSION['username'] == $result['author"])X{ 
让 如 果 是 ， 就 将 $bool 设 为 true*/ 
$bool = true; 
?> 
4- 同时 显示 “删除 ”按钮 ， 并 将 文章 id 作为 url 后 缀 一 起 传 到 处 理 页 -一 一 一 一 一 一 一 > 








<a href="del_file.php?file_id=<?php echo $result['id'];?>"><img src="images/A_delete.gif' width="52" height= 
"16" alt=" 删 除 博客 文章 " onClick="return d_chk();"></a> 
<?php } ?> 
<?php 
ee /分 页 代码 部 分 略 
$sql=mysqli_query(Slink,"select * from tb_filecomment where fileid='$file_id' order by id desc limit $offset, 
$page_size"); 
$result=mysqli_fetch_array($sql); 
exexskkwxwttwhshwtwsxsssystsssssts 根 据 反问 记录 集 输出 记录 *essesssstssstssxsssattatraswasestl 
if$result==false){ 
echo "<tr><td align='center><font color=#ff0000> 对 不 起 ， 没 有 相关 评论 !</font></td></tr>"; 
Jelse{ 

do{ 


?> 
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<tr> 
<td align="center" valign="top" ><table width="666” border="0" cellpadding="5" 
cellspacing="1" bordercolor="#D6E7A5" bgcolor="#FFFFFF" class="i table"> 
<tr bgcolor="#FFFFFF" height="30"> 
<td width="14%" align="center"> 评 论 ID 号 </td> 
< 一 一 一 一 一 一 一 评论 id- 一 一 一 一 一 一 一 一 
<td width="15%"><?php echo $result['id]; ?></td> 
<td width="11%" align="center"> 评 论 人 </td> 
< -一 一 一 一 一 一 一 一 评论 人 -一 一 一 一 一 一 -一 一 
<td width="18%"><?php echo $result['username]; ?></td> 
<td width="12%" align="center"> 评 论 时 间 </td> 
< 上 一 一 一 一 一 一 一 一 评论 时 间 -- 一 一 一 一 一 -一 -> 
<td width="20%"><?php echo $result['"datetime']; ?></td> 
</tr> 
<tr bgcolor="#FFFFFF"> 
<td align="center"> 评 论 内 容 </td> 
< 一 一 一 一 一 一 一 一 一 评论 内 容 一 一 一 一 一 > 
<td colspan="4" style="line-height:1.5">&nbsp;&nbsp;&nbsp;&nbsp;<?php echo Sresult 
[content]; ?></td> 
<td> 
<?php 
让 如 果 $bool 为 真 ， 输 出 删除 超 链接 ， 否 则 跳 过 */ 
if ($bool}{ 
?> 


<a href="del_comment.php?comment_id=<?php echo $result['id]?>"><img src="images/A_delete. 
gif" width="52" height="16" alt=" 删 除 博客 文章 评论 " onClick="return d_chk();"></a> 
<?php }?> 
</td></tr> </table></td> </tr> 
<?php 

}while($result=mysqli_fetch_array($sql)); 
} 
?> 
</table> 
“» <!-- 翻 页 功能 代码 部 分 略 --> 


7.6.6 ”删除 文章 、 评 论 的 实现 过 程 


国 ”删除 文章 、 评 论 模块 使 用 的 数据 表 : tb_article、tb_filecomment 

在 查看 文章 评论 页 面 ， 当 系统 判定 当前 用 户 为 管理 员 或 文章 作者 时 ， 在 每 篇 文章 和 评论 的 后 面 ， 
都 将 显示 相应 的 “删除 ”按钮 。 单 击 任意 的 “删除 ”按钮 ， 系 统 会 提示 是 否 删除 ， 如 果 确 认 ， 将 跳 转 
到 处 理 页 (del_file.php 和 del commentphp)， 完 成 删除 操作 。 

在 删除 文章 的 处 理 页 中 ， 删 除 文章 的 同时 ， 也 删除 了 该 篇 文章 相关 的 评论 。 处 理 页 首先 在 文章 列 
表 (tb_article) 中 删除 id 等 于 $file id 的 记录 ， 如 果 没 有 可 删除 记录 ， 则 提示 失败 ， 并 返回 上 一 步 ， 如 
果 删 除 成 功 , 则 转 到 评论 列表 (tb_filecomment) 中 , 删除 所 有 该 篇 文章 的 评论 。 删 除 文章 页 (del_filephp) 
的 关键 代码 如 下 : 
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倒 程 05 ”代码 位 置 ， 资源 包 \TM\07\tmlog\del file.php 





<?php 
header("Content-type: text/html; charset=utf-8"); /设置 文件 编码 格式 
session_start(); /开启 session 支持 
include "check_login.php"; /包含 权限 检查 文件 
include "Conn/conn.php"; /包含 数据 库 链接 文件 
$sql="delete from tb_article where id=".$_GET[file_id]; /删除 文章 的 delete 语句 
$result=mysqli_query($link,$sql); /执行 删除 操作 
if($resultY{ 
$sql1 = "delete from tb_filecomment where fileid = "$_GET[file_id]; // 删 除 相 对 应 的 文章 评论 
$rst1 = mysqli_query(S$link,$sqI1); /执行 删除 评论 的 操作 
if($rst1) 
echo "<script>alert(' 博 客 文章 已 被 删除 ");location='myfiles.php';</script>"; 
else 
echo "<script>alert(' 删 除 失败 "");history.go(-1);</script>"; 
} 
else{ 


echo "<script>alert( 博 客 文章 删除 操作 失败 !");history.go(-1);</script>"; 
} 


?> 








删除 文章 的 过 程 也 包含 了 删除 评论 的 过 程 ， 所 以 这 里 就 不 给 出 删除 评论 的 代码 了 。 


7.7 图 片上 传 模 块 设计 






We 
由 于 动态 网 络 编程 技术 的 诞生 ， 使 得 网 络 更 加 人 性 化 ， 为 了 能 够 和 用 户 更 好 地 互动 ， 很 多 网 站 都 
提供 了 让 用 户 上 传 图 片 的 功能 。 一 个 网 站 拥有 图 片上 传 功 能 是 非常 必要 的 。 


7.7.1 图 片上 传 模块 概述 


图 片上 传 在 动态 网 页 开发 过 程 中 应 用 非 
常 广泛 。 如 果 有 比较 好 的 图 片 想 和 其 他 人 一 同 
分 享 , 就 可 以 通过 图 片上 传 功能 来 实现 , 以 增 
加 网 站 的 核心 竞争 力 。 本 系统 的 图 片上 传 模块 
主要 实现 对 图 片 的 添加 、 浏览 、 查 询 和 删除 操 
作 ， 而 对 图 片 的 删除 则 只 有 管理 员 才 有 权限 。 
图 片上 传 模块 框架 图 如 图 7.18 所 示 。 


7.7.2 ”图片 上传 模 块 技术 分 析 


既然 是 图 片上 传 模块 ， 显 而 易 见 本 节 的 / 
主要 技术 就 是 上 传 图 片 功能 了 。 上 传 图 片 和 。 区 用 mi 
上 传 文件 的 原理 基本 相同 ， 下 面 就 来 学 习 如 7.18 图 片上 传 模块 框架 图 
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何 上 传 图 片 和 图 片 的 两 种 保存 方式 。 
1. 上 传 图 片 的 基本 流程 
在 网 页 中 实现 上 传 图 片 功能 的 步骤 如 下 : 
(1) 通过 <form> 表 单 中 的 file 元 素 选取 上 传 数据 。 
使 用 fne 元 素 上 传 数 据 时 注意 一 点 : 就 是 在 form 表单 中 要 加 上 属性 enctype="multipart/form-data"， 


否则 上 传 不 了 文件 图片 )。 
(2) 在 处 理 页 中 使 用 $_FILES 变量 中 的 属性 判断 上 传 文件 类 型 和 上 传 文件 图片 大 小 是 否 符合 


要 求 。 
$_FILES 变量 为 系统 预定 义 变量 ， 保 存 的 是 上 传 文件 《图片 ) 的 相关 属性 。 使 用 格式 如 下 : 


$_FILES[name][property]; 


$_FILES 的 相关 属性 如 表 7.13 所 示 。 
表 7.13 $_FILES 的 相关 属性 


属 性 值 说 了 明 
name 上 传 文件 的 文件 名 
type 上 传 文件 的 类 型 
Size 上 传 文件 的 大 小 
tmp_name 上 传 文件 在 服务 器 中 的 临时 文件 名 
erTOT 上 传 文件 失败 的 错误 代码 
(3) 使 用 move_uploaded file0 函 数 上 传 文件 〈 图 片 ) 或 将 文件 〈 图 片 ) 以 二 进 制 的 形式 保存 到 数 


据 库 中 。 

使 用 函数 将 文件 (图片) 保存 到 对 应 的 文件 夹 中 和 以 二 进 制 的 形式 保存 到 数据 库 中 是 上 传 文件 (图 

片 ) 的 两 种 形式 ， 稍 后 将 单独 做 介绍 。 
(4) 返回 页 面 等 待 下 一 步 操作 。 

2. 使 用 上 传 函数 保存 文件 图片) 

使 用 上 传 函 数 上 传 文件 〈 图 片 ) 的 本 质 就 是 将 文件 〈 图 片 ) 从 浏览 器 端 复制 到 服务 器 端 指定 的 
文件 夹 中 ， 数 据 库 所 存储 的 就 是 文件 《图片 ) 的 相对 地 址 。 当 页 面 显 示 图 片 时 ， 实 际 是 分 两 步 : 第 
一 步 是 读 取 数据 表 中 的 地 址 ， 第 二 步 是 根据 地 址 找到 并 在 页 面 中 显示 图 片 。 使 用 目录 保存 文件 的 好 处 
是 减少 了 数据 库 的 容量 和 对 数据 库 的 压力 ， 而 且 图 片 很 容易 被 搜索 引擎 抓 到 ， 从 而 提高 网 站 的 流量 和 
A 

move_uploaded_file(0) 函 数 的 一 般 格式 如 下 : 








bool move_uploaded file(string filename, string destination); 


回 ”人 lename: 上 传 到 服务 器 中 的 临时 文件 名 。 
回 destination: 保存 文件 的 实际 路 径 。 
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sa 
[tmp_name'] 来 获取 。 


下 面 看 一 个 实例 ， 程 序 代码 如 下 : 


<label> 请 选择 要 上 传 的 图 片 ( 文 件 小 于 500K): </label> 

<!-- 上 传 文件 的 form 表单 -> 

<form method="post" action="#" enctype="multipart/form-data"> 
<input type="hidden" name="action" value="upload" /> 
<input type="file" name="u_file"/> 
<input type="submit" value=" 上 传 " /> 


</form> 
a 
<?php 
* 判 断 是 否 为 上 传动 作 */ 
if(isset($_POST[action])X 
Sfile_path = "./"; /上 传 文件 存放 路 径 ，./ 为 当前 目录 下 
$profix = array(".jpg",".gif"," .jpeg"); /| 设置 允 许 上 传 的 文件 后 缀 类 型 
$f_name = $_FILES[u_filel[name']; // 取 得 要 上 传 的 文件 名 
S$pro_name=substr($f_name,strrpos($f_name,".")); 1/ 取得 上 传 文件 的 后 缀 
if(!in_array($pro_name, $profix)X{ /| 判断 上 传 文件 的 类 型 是 否 为 允许 类 型 
echo "文件 格式 不 对 "; 
exit(); 
} 
if($_FILES['u_file"['size'] >= 500000X( // 判 断 上 传 文件 的 大 小 
echo "文件 上 传 错误 ， 或 文件 大 于 500KB, 请 重新 上 传 "” ”// 如 果 文件 过 大 ， 提 示 错 误 
} 
else{ 
人 * 上 传 文件 函数 */ 
move_uploaded _file($_FILES['u_file [tmp_name'],$file_path.$_FILES['u_file [name']); 
echo "上 传 成 功 "; 


?> 





3. 使 用 二 进 制 保存 图 片 


上 传 图 片 的 另 一 种 保存 方式 是 以 二 进 制 的 形式 存储 在 数据 库 中 。 在 计算 机 看 来 ， 再 美的 图 片 、 再 感人 
的 文章 ， 也 不 过 就 是 一 堆 “0111010011…” 的 代码 段 而 已 ， 和 其 他 数据 没有 什么 区 别 ， 存 到 数据 表 中 都 是 
一 样 的 。 存 储 图 片 不 再 需要 使 用 move_uploads_file0 函 数 ， 在 处 理 页 中 直接 使 用 二 进 制 的 形式 读 取出 文件 
图 片 ， 并 存 到 数据 表 中 。 本 系统 采用 的 就 是 以 二 进 制 的 形式 来 保存 图 片 ， 具 体操 作 将 在 实现 过 程 中 给 出 。 

使 用 二 进 制 来 保存 图 片 的 好 处 是 安全 ,特别 是 涉及 个 人 隐私 ， 不 易 被 窃取 ;存储 方便 ， 和 表 中 其 
他 数据 同等 操作 就 可 以 ， 节 省 磁盘 空间 ， 由 于 文件 系统 类 型 的 限制 ， 放 到 数据 表 中 的 数据 一 定 比 直接 
放 到 磁盘 中 要 节省 空间 。 

当然 二 进 制 形式 的 图 片 也 有 很 大 浆 端 ， 在 传 出 的 过 程 中 ， 如 果 某 个 二 进 制 位 出 现 丢 失 或 损坏 ， 那 
么 整 张 图 片 将 无 法 显示 ， 而 是 变 为 一 堆 乱码 ， 而 且 如 果 存 储 的 图 片 过 多 ， 造 成 数据 库 过 于 庞大 ， 将 影 
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响 其 他 数据 的 正常 读 取 ， 增 大 服务 器 的 负担 。 所 以 ， 对 于 太 大 的 文件 ， 就 不 适宜 保存 到 数据 库 中 ， 而 
是 应 该 放 到 文件 夹 中 。 


7.7.3 图 片上 传 的 实现 过 程 


国 图片 上传 使 用 的 数据 表 : tb tpsc 

博客 用 户 登录 后 ， 单 击 导航 栏 中 的 “图 片 管 理 ”/“ 添 加 图 片 ”选项 ， 即 可 进入 添加 图 片 页 面 ， 在 
“图 片 名 称 ” 文 本 框 中 添加 上 传 的 图 片 名 称 ， 在 “上 传 路 径 ” 文 本 框 中 选择 或 者 单 击 “浏览 ”按钮 选 
择 自己 喜欢 的 图 片 ， 单 击 “ 提 交 ” 按 钮 ， 以 二 进 制 形式 将 图 片上 传 到 数据 库 中 。 图 片上 传 页 面 的 运行 
结果 如 图 7.19 所 示 。 











国光 :总 = 
L 上 Me: [EE lonpg |] 


[到 | 





图 7.19 图 片上 传 页 面 的 运行 结果 


图 片上 传 页 是 一 个 上 传 文件 的 表单 ， 主 要 包括 一 个 文本 域 、 一 个 文件 域 和 一 个 “提交 ”按钮 。 部 
分 表单 元 素 的 名 称 及 属性 如 表 7.14 所 示 。 





表 7.14 图 片上 传 页 面 中 的 表单 元 素 


名 称 含义 
myform | tm | method="post" action="tptj_ok.php” enctype="multipart/form-data" 图 片上 传 表单 
tpme ="text" id="tpme" size="40" 图 片 名 称 

file ="file" size="23" maxlength="60" 上 传 路 径 

bm tj id="btn_tj" value=" 提 交 " onClick="return pic_chkO:" “提交 ”按钮 





当 用 户 输入 图 片 名 称 , 并 选择 图 片 路 径 后 , 单 击 “ 提交” 按钮 ,系统 将 进入 到 上 传 处 理 页 (tptj_ok.php) 
中 进行 处 理 。 在 处 理 页 中 ， 首 先 对 图 片 名 称 进行 处 理 ， 去 掉 特 殊 字 符 、 空 行 和 空格 ， 然 后 对 上 传 的 文 
件 进 行 类 型 检查 和 文件 大 小 检查 。 最 后 ， 以 二 进 制 的 形式 和 图 片 的 其 他 信息 〈 如 上 传 用 户 、 上 传 时 间 
等 ) 一 起 存 进 数据 表 中 。 关 键 代码 如 下 : 


倒 程 06 代码 位 置 : 资源 包 \TM\07\tmlog\tpti_ ok.php 


<?php 

header("Content-type: text/html; charset=utf-8"); /1 设置 文件 编码 格式 
session_start(); /开启 session 支持 
include "check_login.php"; // 包 含 权限 检查 文件 
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include "Conn/conn.php"; // 包 含 数据 库 链 接 文件 
date_default_timezone_set("PRC"); /1 设置 时 区 
$tpmc=$_POST[tpmc]; // 获 取 上 传 图 片 名 称 
if($_POST["btn_t)"]==" 提 交 "X{ 
© $tpmc=htmlspecialchars($tpmc); /将 图 片 名 称 中 的 特殊 字符 转换 成 HTML 格式 
@ $tpmc=str_replace("n","<br>",$tpmc): /将 图 片 名 称 中 的 回 车 符 以 自动 换行 符 取代 
$tpmc=str_replace(" ","&nbsp;", $tpmce); /将 图 片 名 称 中 的 空格 以 “&nbsp;” 取 代 
$author=$_SESSION[username']; 
$scsj=date("Y-m-d"); // 设 置 图 片 的 上 传 时 间 
$profix = array(".jpg",".gif",".jpeg"); // 设 置 允 许 上 传 的 文件 后 缀 类 型 
$f_name = $_FILES[file]rname']; // 取 得 要 上 传 的 文件 名 


S$pro_name=substr($f_name,strrpos($f_name,"."));，// 取 得 上 传 文件 的 后 缀 
/判断 上 传 文件 的 类 型 是 否 为 允许 类 型 */ 
© if(!in_array(strtolower($pro_name), $profix)}{ 
echo "<script>alert(' 文 件 格式 不 对 '");history.go(-1);</script>"; 
exit(); 


3: 

/判断 上 传 文件 的 大 小 ， 如 果 文 件 过 大 ， 提 示 错 误 */ 

if$_FILES[filel[size] > 500000){ 
echo "<script>alert(' 文 件 上 传 错误 ,请 重新 上 传 ');history go(-1)</script>"; 
exit(); 


$fp=fopen($_FILES['file [tmp_name’],"r"); /以 只 读 方式 打开 文件 
$file=addslashes(fread($fp,filesize($_FILES[file]rtmp_name]))) /将 文件 中 的 引号 部 分 加 上 反 斜 线 
$query="insert into tb_tpsc (tpmc,file,author scsj) values ('$tpmce','$file','$author','$scsj')"; 
$result=mysqli_query($link,$query); 
echo "<meta http-equiv=\"refresh\" content=\"1;url=browse_pic.php\"> 图 片上 传 成 功 ， 请 稍 等 …"; 

} 


?> 





Ah 代码 贴 二 
@ htmlspecialcharsO) 函 数 : 将 特殊 字符 转换 成 HTML 格式 。 
@ str_ replace() 函 数 : 取代 所 有 在 字 事 中 出 现 的 字 事 。 语 法 如 下 : 
mixed str_replace(mixed search, mixed replace, mixed subject, int &count) 


str_TeplaceO 函 数 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 Teplace 替换 ,参数 &count 表 示 替 换 字符 串 执行 的 次 数 。 
日 strtolower: 将 字符 转换 为 小 写字 母 。 


7.7.4 图 片 浏览 的 实现 过 程 


国 ” 图片 浏 览 使 用 的 数据 表 : tb tpsc 

无 论 是 注册 用 户 ， 还 是 非 注 册 用 户 ， 只 要 登录 网 站 ， 就 可 以 无 条 件 地 浏览 所 有 图 片 。 而 删除 图 片 
除了 管理 员 ， 其 他 人 都 无 权 操 作 。 非 注册 用 户 可 以 通过 首页 中 的 “最 新 图 片 ” 进 入 图 片 浏览 页 面 ， 注 
册 用 户 先 进入 个 人 管理 界面 ， 单 击 “ 图 片 管理 ”/“ 浏 览 图 片 ”菜单 ， 同 样 可 以 进入 图 片 浏览 页 面 。 注 
册 用 户 浏 览 图 片 页 面 的 运行 结果 如 图 7.20 所 示 。 


@ 
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CE 


图 7.20 浏览 图 片 页 面 的 运行 结果 


本 页 的 实现 代码 和 查看 文章 页 面 略 有 不 同 ， 查 看 文章 页 面 中 ， 每 条 数据 占 了 一 行 ， 而 查看 图 片 则 
采用 的 是 分 栏 显示 ， 以 每 行 两 张 图 片 的 格式 输出 ， 每 页 显示 4 张 图 片 。 通 过 单 击 小 图 片 ， 可 以 查看 图 
片 原 效果 图 。 程 序 关键 代码 如 下 : 


倒 程 07 ”代码 位 置 ， 资源 包 \TM\07\tmlog\browse_pic.php 





<?php 
session_start(); /开启 session 支持 
include "Conn/conn.php"; // 包 含 数据 库 链 接 文 件 
include "check_login.php"; // 包 含 权限 检查 文件 

?> 

人 <!- 首 部 导航 栏 -> 

<tr align="|left"> 


<td height="25" valign="top” bgcolor="#EFF7DE"> <span class="tableBorder_ LTR"> 浏 览 图片 </span> 
</td></tr> 
<tr> 
<td height="192" align="center valign="top" > 
<?php 
~- // 分 页 显示 
$query="select * from tb_tpsc where scsj order by id desc limit $offset, $page_size"; 
让 返回 结果 集 */ 
Sresult=mysqli_query($link, $query); 
Pe 
<table width="666" border="0" align="center" cellpadding="3" cellspacing="1" bordercolor="#D6D7D6"> 
<tr> 
<?php 
让 设置 一 个 变量 ， 判 断 当 前 数据 是 奇数 输出 还 是 偶数 输出 */ 
$i=1; 
while($info=mysqli_fetch_array($result) 
{ 


辐 
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* 如 果 当 前 为 偶数 输出 ， 记 录 输 出 完毕 后 结束 <tr> 标 签 */ 
if($i%2==0X{ 
?> 
<td height="290"><table class="i table" align="center" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td colspan="2"> 
< 上 -显示 图 片 页 image.php 一 > 
<a href="image.php?recid=<?php echo $info[id]; ?>" target="_blank"><img src= 
"image.php?recid=<?php echo $info[id]:?>" width="300" height="230"></a></td></tr> 
<tr> 
<td width="150” height="23"”align="left">&nbsp; 图 片 名 称 :<?php echo S$info 
[tpmc];?> </td> 
<td colspan="2" align="right"> 
<?php 
上 * 如 果 登 录用 户 为 管理 员 ， 显 示人 删除 操作 */ 
if ($_SESSION[fig]==1X{ 
?> 
<a href="remove.php?pic_id=<?php echo S$info[id]?>"><img src="images/ 
A_delete.gif" width="52" height="16" alt=" 删 除 图 片 " onClick="return pic_chk();"></a> 


<?php 
} 
?> 
</td> 
</tr> 
<tr> 
<td width="150" height="20">&nbsp; 上 传 时 间 : <?php echo $info['scsj];?></td> 
</tr> 
</table></td> 
</tr> 
<?php 
否则 ， 开 始 新 的 一 行 */ 
else 
{ 
?> 
<td><table class="i_table" align="center" width="236" border="0" cellpadding="0" cellspacing="0"> 


<tr> 
<td colspan="2"><a href="image.php?recid=<?php echo S$info[id']; ?>" target="_blank"> <img src="image. 
php?recid=<?php echo $info['id];?>" width="300" height="230"></a></td> 
</tr> 
<tr> 
<td width="150" height="23">&nbsp; 图 片 名 称 :<?php echo $info[tpmc]?></td> 


<td colspan="2" align="right"> 
<?php 
if ($_SESSION['fig]==1){ 
?> 
<a href="remove.php?pic_id=<?php echo $info[id]?>"><img src="images/ 
A_delete.gif" width="52" height="16" alt=" 删 除 图 片 " onClick="return pic_chk();"></a> 


全 
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<?php 
} 
?> 
</td> 
</tr> 
<tr> 
<td width="150" height="20">&nbsp; 上 传 时 间 : <?php echo Sinfo['scsj];?></td> 
</tr> 
</table></td> 

<?php 


J 
/变量 $i 自 加 1，while 循环 结束 */ 
$it+; 


} 


> 
</tr> 


通过 上 述 代码 可 以 看 到 ,预览 图 片 和 显示 图 片 实际 调用 的 都 是 同一 页 面 : image.php。image.php 页 
就 是 用 来 显示 图 片 的 实际 效果 的 。 实 现代 码 如 下 : 


倒 程 08 代码 位 置 ， 资源 包 \TM\07\tmlog\image.php 





<?php 
header("Content-type: image/png"); // 设 置 输出 为 图 片 格式 
include "Conn/conn.php"; // 包 含 数据 库 链接 文件 
$query="select * from tb_tpsc where id=".$_GET[recid]; /根据 id 生成 查询 语句 
Sresult=mysqli_query(S$link, $query); /执行 查询 语句 
if(I$result) die("error: mysqli query"); // 判 断 是 否 有 返回 结果 
Snum=mysqli_num_rows($result); // 计 算 结 果 集 个 数 
if($num<1) die("error: no this recorder"); 
$data = mysqli_fetch_array( $result); // 返 回 图 片 数据 
echo $data[file]; // 输 出 图 片 
ca 





7.7.5 删除 图 片 的 实现 过 程 


国 。 删除 图 片 使 用 的 数据 表 : tb tpsc 

删除 图 片 是 管理 员 才 有 的 权限 ， 在 图 片 浏览 的 实现 代码 中 ， 已 经 给 出 如 何 判断 登录 用 户 是 否 为 管 
理 员 。 当 管理 员 单 击 “ 删 除 ” 超 链接 时 ， 处 理 页 (remove.php) 会 根据 传 过 来 的 id， 删除 对 应 的 数据 
表 中 的 数据 。 实 现代 码 如 下 : 


倒 程 09 代码 位 置 资源 包 \TM\07\tmlog\remove.php 


<?php 
header("Content-type: text/html; charset=utf-8"); // 设 置 文 件 编码 格式 
include "Conn/conn.php"; // 包 含 数据 库 链接 文件 
$sql="delete from tb_tpsc where id="$_GET[pic_id]; /生成 删除 语句 
$result=mysqli_query($link,$sql): /执行 删除 语句 
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该 $result){ /根据 $result 返回 结果 
echo "<script>alert(' 图 片 删除 成 功 ");location='browse_pic.php';</script>"; 

} 

else{ 
echo "<script>alert(' 图 片 删除 操作 失败 !");history.go(-1);</script>"; 

} 


?> 








7.8 朋友 圈 模 块 设计 


7.8.1 ”朋友 轿 模 块 概述 


本 系统 的 朋友 圈 模 块 的 主要 功能 是 添加 、 查 询 、 删 除 好 友 ， 添 加 的 好 友 除了 该 用 户 以 外 ， 包 括 管 
理 员 在 内 的 所 有 外 人 都 不 可 以 查看 ， 以 保证 其 个 人 隐私 不 被 外 泄 。 用 户 被 删除 时 ， 该 用 户 现 有 的 朋友 
圈 也 一 并 被 删除 。 朋 友 圈 模块 框架 图 如 图 7.21 所 示 。 


7.8.2 朋友 轿 模 块 技术 分 析 





































查询 好 友 ”| 

在 查询 好 友 的 功能 中 ， 使 用 到 了 模糊 查询 语句 ， 可 看 好 友 | | 

用 于 模糊 查找 好 友 列表 。 模 糊 查 询 语句 使 用 的 是 like re 
运算 符 。 在 PHP 中 , 带 有 like 运算 符 的 查询 语句 的 党 | 
用 格式 有 以 下 两 种 : | 








(1) 使 用 通配符 “%” 的 where 子 句 。 


通配符 “%” 表 示 0 个 或 多 个 任意 长 度 和 类 型 的 
字符 ， 包 括 中 文 汉字 。 图 7.21 朋友 圈 模 块 框架 图 
示例 1: 表示 查找 所 有 内 容 包 含 “ 好 ” 字 的 文章 。 


select * from tb_file where content like '% 好 %'; 
示例 2: 查找 所 有 包含 “好 ” 字 或 “高 ” 字 的 文章 ， 这 时 
select * from tb_file where content like '% 好 %' or content like '% 高 
(2) 使 用 通配符 “_” 的 where 子 句 。 


通配符 “_” 表 示 匹 配 任意 的 单个 字符 。 
示例 1: 查找 用 户 名 只 包含 5 个 字符 ， 其 中 后 4 个 字符 为 








select * from tb_user where regname like '_soft'; 
示例 2: 查找 所 有 以 + 开头， 并且 以 + 结 尾 的 、 中 间 包 含 : 


select * from tb_user where regname like tt _ ft 








324 
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查找 的 结果 为 : tsoft。 


人 注意 公用 MysQL 亿 模 机 坦 询 要 注 总 编码 问题 如果 编码 不 统 _， 天 么 查询 时 就 安 易 查 不 到 吉 
据 ， 或 返回 的 数据 不 匹配 。 所 以 在 安装 MySQL 时 ， 要 保持 和 系统 编码 的 统一 。 常 用 的 编码 格式 有 
gb2312、ISO-8859-1、utf8 和 gbk 等 。 


7.8.3 添加 好 友 的 实现 过 程 


国 添加 好 友 使 用 的 数据 表 : tb_friend 

博客 用 户 登 录 后 ， 单 击 “ 朋 友 圈 管理 ”/“ 添 加 到 朋友 圈 ”， 会 进入 添加 好 友 页 面 。 在 表单 中 输入 
要 添加 的 好 友信 息 ， 单 击 “ 提 交 ” 按 钮 ， 系 统 会 跳 转 到 处 理 页 进行 处 理 。 添 加 好 友 页 面 的 运行 结果 如 
7.22 所 示 。 








王仁 二 村 兴 贡 
MEA 








TD 
图 7.22 添加 好 友 页 面 的 运行 结果 
添加 好 友 页 面包 含 一 个 用 户 输入 表单 ， 主 要 表单 元 素 如 表 7.15 所 示 。 


表 7.15 添加 好 友 页 面 表单 的 主要 元 素 属 性 






method="post" action="check friend.php" 添加 好 友 表 单 
id="txt_name" size=20 好 友 姓 名 
<option value=1> 男 </option> 

txt_sex select <option value=2> 女 </option> 好 友 性 别 
<option value=0 selected> 保 密 </option> 

txt_bir text id="txt_bir" 好 友 出 生日 期 
txt_province select id="select" onChange= "initcityO:” 好 友 所 在 省 份 
txt_city select id="select2" 好 友 所 在 城市 
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续 表 
名 称 元 素 类 型 含义 
txt_address text size="40" 好 友 家 庭 住 址 
txt_postcode text size="40" 邮政 编码 
txt_ email text size="40" 好 友 邮 箱 地 址 
txt_tel text size="20" 好 友 家 庭 电话 
txt_handset text size="20" 好 友 手 机 号 码 
txt QQ text size="20" 好 友 QQ 号 码 
regsubmit submit onClick="retum checkO" “提交 ”按钮 
Submit2 reset class="btn_grey" “ 重 填 ”按钮 





当 处 理 页 接收 到 用 户 输入 的 好 友信 息 后 , 应 用 INSERT 语句 将 好 友信 息 添 加 到 数据 表 tb_friend 中 。 


如 果 添 加 成 功 ， 则 弹出 “该 信息 添加 成 功 !” 的 提示 框 ; 否则 弹出 “添加 操作 失败 !” 的 提示 框 。 处 理 


页 的 关键 代码 如 下 : 
倒 程 10 ”代码 位 置 ， 资源 包 \TMW7\tmlog\check_friend.php 





<?php 


header("Content-type:text/html;charset=utf-8"); 
session_start(); 


include "Conn/conn.php"; 


if($_POST["regsubmit"]<>"™"X{ 
$name=$_POST['txt_name'; 
S$sex=$_POST[txt_sex']; 
$bir=$_POST[txt_bir]; 


$city=$_POST[txt_province].$_POST[txt_city]; 


$address=$_POST[txt_address']; 
S$postcode=$_POSTT'txt_postcode'; 
S$email=$_POST[txt_email]; 


Stel=$_POST[txt tel]: 


$handset=$_POST[txt_handset]; 
$QQ=$_POST[txt_QQ]; 
$username=$_SESSION['username']; 


$INS="Insert Into tb_friend (name,sex,bir,city,address,postcode,email,tel,handset,QQ,username) Values 


// 设 置 文 件 编码 格式 
/开启 session 支持 


// 包 含 数据 库 链接 文件 
// 如 果 用 户 单 击 了 “提交 ”按钮 


// 获 取 好 友 姓名 

// 获 取 好 友 性 别 

// 获 取 好 友 生 日 

// 获 取 好 友 所 在 城市 
// 获 取 好 友 家 庭 住 址 
// 获 取 邮 政 编码 

// 获 取 好 友 邮 箱 地 址 
// 获 取 好 友 家 庭 电 话 
// 获 取 好 友 手 机 号 码 
// 获 取 好 友 QQ 号 码 
// 获 取 登 录用 户 名 


(‘$name','$sex','$bir','$city','$address','$postcode','$email','$tel','$handset','$QQ','$usermmame')";// 定 义 插入 语句 
Sresult=mysqli_query($link, $INS); 
if($resultY{ 


} 


else{ 
echo "<script> alert(' 添 加 操作 失败 ! ");</script>"; 
echo "<script> window.location='friend.php';</script>"; 


echo "<script> alert(' 该 信息 添加 成 功 ! ');</script>"; 
echo "<script> window.location='friend.php';</script>"; 


/执行 插入 语句 
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7.8.4 浏览 好 友 的 实现 过 程 


国 查询 好 友 使 用 的 数据 表 : tb friend 
博客 用 户 登 录 后 ， 单 击 “ 朋 友 圈 管理 ”/“ 浏 览 我 的 朋友 ”， 进 入 浏览 好 友 页 面 。 在 该 页 面 中 会 显 
示 该 用 户 所 有 好 友 的 详细 信息 。 浏 览 好 友 页 面 的 运行 结果 如 图 7.23 所 示 。 
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图 7.23 浏览 好 友 页 面 的 运行 结果 


在 浏览 好 友 页 面 ， 应 用 SELECT 语句 对 登录 用 户 的 所 有 好 友 进 行 查询 ， 并 对 查询 结果 进行 分 页 输 
出 。 浏 览 好 友 页 面 的 关键 代码 如 下 : 


倒 程 11 代码 位 置 ， 资源 包 \TM\07\tmlog\browse_friphp 


<?php 
session_start(); /开启 session 支持 
include "Conn/conn.php"; /包含 数据 库 链 接 文件 
include "check_login.php"; /包含 权限 检查 文件 
?> 
<?php 
$page_size=2; /每 页 显示 2 条 记录 
$query="select count(*) as total from tb_friend where usemame=".$_SESSION['username']." order by id 
desc", 
Sresult=mysqli_query($link, $query); /查询 总 的 记录 条 数 
$data = mysqli_fetch_array($result):; 
$message_count=$data[total]; /获取 查询 总 记录 数 
$page_count=ceil($message_count/$page_size); 。 // 根 据 记录 总 数 除 以 每 页 显示 的 记录 数 求 出 所 分 的 页 数 
Soffset=($page-1)*$page_size; /计算 下 一 页 从 第 几 条 数据 开始 循环 


$sql=mysqli_query($link,"select * from tb_friend where username=".$_SESSION['username']." order by id 
desc limit $offset, $page_size"); 
$result=mysqli_fetch_array($sql); 
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if(I$result}{ 
?> 
<tr><td align="center"><font color=#ff0000> 您 还 未 添加 任何 朋友 !</font></td></tr> 


<?php 
Jelse{ 
dof 
?> 
<!- 显 示 好 友 详 细 详 细 -> 
<?php 


}while($result=mysqli_fetch_array($sql)); 
} 


?> 





7.8.5 查询 好 友 的 实现 过 程 


国 查询 好 友 使 用 的 数据 表 : tb friend 
当 用 户 要 查询 好 友 时 ， 单 击 “ 朋 友 圈 管理 ”/“ 查 询 朋 友信 息 ”， 显 示 查 询 页 面 。 查 询 可 以 分 为 姓 
名 查询 和 编号 查询 ， 均 为 模糊 查询 。 当 用 户 输入 要 查找 的 关键 字 后 ， 单 击 “ 检 索 ” 按 钮 ， 或 按 Enter 
键 ， 系 统 跳 到 处 理 页 进行 处 理 。 查 询 好 友 页 面 的 运行 结果 如 图 7.24 所 示 。 
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图 7.24 查询 好 友 页 面 的 运行 结果 


查询 页 包含 一 个 查询 表单 ， 包 括 查询 条 件 和 查询 关键 字 两 部 分 表单 元 素 。 主 要 表单 元 素 如 表 7.16 
所 示 。 


表 7.16 查询 页 表单 的 主要 元 素 属性 


<option value="name" selected> 姓 名 </option> 


<option value="id"> 编 号 </option> 
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当 处 理 页 接收 到 查询 条 件 及 查询 关键 字 后 ， 生 成 模糊 查询 语句 ， 执 行 SQL 语句 并 返回 查询 结果 。 
如 果 没 有 输入 关键 字 , 则 弹出 提示 框 ; 如 果 没 有 查找 到 任何 结果 ， 则 输出 “Sorry! 没 有 您 要 找 的 朋友 !”。 
处 理 页 的 关键 代码 如 下 : 


倒 程 12 代码 位 置 ， 资源 包 \TM\07\tmlog\query_friend.php 


<?php 

session_start(); /开启 session 支持 
include "Conn/conn.php"; /包含 数 据 库 链接 文件 
include "check_login.php"; /包含 权限 检查 文件 


?> 
<script language="javascript"> 
/JavaScript 脚本 ， 验 证 输入 信息 ， 如 果 为 空 ， 则 弹出 提示 框 */ 
function check(form}{ 
if (document.myform.sel_key.value=="™"){ 


alert(" 请 输入 查询 条 件 ""); 
myform.sel_keyfocus(); 
return false; 
} 
} 
</script> 
<?php 
if (isset($_POST["Submit]) && $_POST["Submit"]==" 检 索 "){ 
$tj=$_POST['sel_t)]; // 接 收 查询 字段 
$key=$_POST['sel_key'; /接收 查询 关键 字 


$sql=mysqli_query($link,"select * from tb_friend where $tj like '%$key%' and usemame=".$_SESSION 
[username].""); 


$result=mysqli_fetch_array($sql); /执行 查询 语句 
if($result==false){ 

echo ("[<font color=red>Sorry! 没 有 您 要 找 的 朋友 !</font>]"); 
}else{ 


?> 
~“ <!- 显 示 记录 表单 -> 
<?php}?> 





7.9 开发 技巧 与 难点 分 析 


想象 一 下 : 动态 的 下 拉 菜 单 ， 半 透明 的 背景 效果 ， 鼠 标 划 过 时 的 色彩 交替 ， 给 人 一 种 亦 真 亦 幻 
的 视觉 冲击 和 美的 感受 ， 这 样 的 网 站 是 不 是 具有 更 大 的 吸引 力 ? 本 系统 个 人 管理 页 面 中 的 管理 菜单 
实现 的 就 是 这 种 效果 。 当 用 户 登录 到 博客 网 站 后 ， 将 鼠标 移动 到 “朋友 圈 管 理 ” 等 导航 链接 上 时 ， 
将 在 下 方 显示 出 半 透 明 的 下 拉 式 菜单 ， 透 过 此 下 拉 菜 单 仍 可 以 看 到 页 面 上 的 内 容 ， 运 行 效果 如 图 7.25 


所 示 。 
四 


PHP 项 目 开发 全 程 实录 (第 4 版 ) 





EE | 





1 、 纺 二 多彩 的 图 形 图 像 寻 理 卸 
2、 百 这 恬 方 -关于 字符 中 的 作 
末 训 霹 大 拖 棕 一 天 | 皇 库 芝 得 技术 
4、PHPE 绩 只 规 共 科 度 重 不 重要 
5、 泛 淡 PHP 的 二 昌 生 
6、PHPE 拔 夫 型 
7、 世 寞 和 外 杂 ,世界 根 入 里 
8、 芝 展 每 一 个 现在 

页 Ri 1/1 页 记录 : 8 条 








图 7.25 动态 半 透 明 效果 的 下 拉 菜 单 


实现 半 透 明 背 景 的 下 拉 菜 单 ， 首 先 需要 在 页 面 中 实现 下 拉 菜 单 ， 然 后 再 通过 设置 下 拉 菜 单 的 CSS 
样式 实现 半 透 明 效果 。 实 现下 拉 菜 单 的 半 透 明 效 果 可 以 应 用 CSS 样式 的 透明 效果 滤 镜 alpha 实现 。alpha 
属性 是 把 一 个 目标 元 素 与 背景 混合 。 这 种 “与 背景 混合 ”， 通 俗 地 说 ， 是 一 个 元 素 的 透明 度 。 透 明 效果 
滤 镜 alpha 的 语法 如 下 : 





{filter:alpha(opacity=opacity,finishopacity=finishopacity, style=style, startx=startx, starty=starty,finishx=finishx,finishy 
=finishy)} 





滤 镜 alpha 的 各 个 参数 说 明 如 表 7.17 所 示 。 


表 7.17 alpha 滤 镜 的 各 个 参数 及 说 明 
属 性 说 了 明 
代表 透明 度 水 准 。 默 认 的 范围 是 0 一 100， 其 实 是 百分比 的 形式 ， 也 就 是 0 代表 完全 透明 ，100 代表 完 
py 全 不 透明 
finishopacity | 可 选 ， 如 果 想 要 设置 渐变 的 透明 效果 ， 可 以 使 用 该 参数 指定 结束 时 的 透明 度 。 范 围 也 是 0~100 














style 指定 透明 区 域 的 形状 特征 ， 其 中 0 代表 统一 形状 、1 代表 线形 、2 代表 放射 状 、3 代表 长 方形 
startx 代表 渐变 透明 效果 的 开始 和 坐标 
starty 代表 渐变 透明 效果 的 开始 立 坐标 
finishx 代表 渐变 透明 效果 的 结束 和 坐标 
finishy 代表 渐变 透明 效果 的 结束 立 坐 标 


在 本 实例 中 的 menuskin 类 中 ， 设 置 了 alpha 滤 镜 的 属性 。 代 码 如 下 : 
倒 程 13 代码 位 置 资源 包 \TM\07\tmlog\ess\style.css 


.menuskin { 
BORDER: #666666 1px solid; / 层 边框 样式 
VISIBILITY: hidden: / 层 可 见 为 隐藏 
FONT 12px Verdana: / 层 中 的 字体 样式 和 大 小 


局 
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POSITION: absolute; /定位 方式 
background-image:url("../images/item_out.gif"); /背景 图 片 

background-repeat : repeat-y; /图 片 是 否 可 重复 

filter: alpha(Opacity=85); /设置 alpha 的 opacity 属性 等 于 85， 即 透明 度 为 85% 


} 
在 显示 页 面 中 显示 菜单 项 的 表单 引入 这 个 menuskin 类 ， 当 鼠标 划 过 特定 的 文字 时 ， 将 会 发 现 自动 





下 拉 菜 单 的 效果 为 半 透 明 的 样式 。 页 面 关键 代码 如 下 : 


倒 竹 14 ”代码 位 置 : 资源 包 \TM\07\tmlog\file.php 
<!-- 弹 出 层 设置 -> 


<div class=menuskin id=popmenu 
<!-- 鼠 标 进入 该 区 域 时 调用 的 js 方法 --> 
onmouseover="clearhidemenu();highlightmenu(event,'on')" 
<!-- 鼠 标 离开 该 区 域 时 调用 的 js 方法 --> 
onmouseout="highlightmenu(event,'off );dynamichide(event)" 
<!-- 设 置 层 全 顺序 和 定位 方式 --> 
style="Z-index:100;position:absolute;"> 

</div> 


<!-- 半 透明 的 动态 下 拉 菜 单 -> 
<TABLE width="650" border="0" cellSpacing=0 cellPadding=0 align="center"> 
<TR align="center valign="middle"> 

<!-- 显 示 登 录用 户 -> 

<TD style="WIDTH: 110px; COLOR: red;"> 欢 迎 您 :&nbsp;<?php echo 
$_SESSION[username']; ?>&nbsp;&nbsp;</TD> 

<!-- 显 示 首页 超 链接 一 > 

<TD style="WIDTH: 90px; COLOR: red;"><SPAN style="FONT-SIZE: 9pt; COLOR: #cc0033"> 
</SPAN><a href="index.php"> 博 客 首页 </a></TD> 

<!- 显 示 文 章 管理 下 拉 菜单 -> 

<TD style="WIDTH: 90px; COLOR: red;"><a onmouseover=showmenu(event,productmenu) 
onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand" > 文章 管理 </a></TD> 

<!- 显 示 图 片 管理 下 拉 菜单 -> 
<TD style="WIDTH: 90px; COLOR: red;"><a onmouseover=showmenu(event,Honourmenu) 
onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand"> 图 片 管理 </a></TD> 

<!-- 显 示 朋 友 轿 管理 下 拉 菜 单 --> 

<TD style="WIDTH: 100px; COLOR: red;"><a onmouseover=showmenu(event,myfriend) 
onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand" > 朋友 圈 管 理 </a></TD> 
<?php 

* 如 果 登 录用 户 为 管理 员 */ 

ifisset($_SESSION[fig]) && $_SESSION[fig]==1){ 
?> 

< 显示 管理 员 管 理 下 拉 菜 单一 一 一 一 一 一 一 一 一 一 > 

<TD style="WIDTH: 90px; COLOR: red;"><a onmouseover=showmenu(event,myuser) 
onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand" > 管理 员 管 理 </a></TD> 
<?php 





} 
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?> 
< 上 一 一 一 一 一 一 一 一 一 一 显示 退出 起 链接 一 一 一 一 一 一 一 一 一 一 > 
<TD style="WIDTH: 90px; COLOR: red;"><a href="safe.php"> 退 出 登录 </a></TD> 
</TR> 
</TABLE> 


通过 代码 可 以 发 现 ， 在 每 个 超 链接 标签 <a> 中 ， 都 有 3 个 参数 ， 分 别 介绍 如 下 。 

回 “onMouseOver: 鼠标 进入 该 区 域 时 的 效果 。 本 页 面 调用 了 js 方法 showmenu()。 

回 onMouseOut: 鼠标 移出 该 区 域 时 的 效果 。 本 页 面 调用 js 方法 delayhidemenu()。 

回 style: 标签 样式 ， 本 页 面 样式 为 “CURSOR:hand;”， 显 示 一 个 手 的 形状 。 

调用 的 js 方法 就 是 实现 下 拉 菜 单 的 JavaScript 代码 : 显示 和 隐藏 菜单 。 由 于 这 段 代码 是 通用 型 代 
码 ， 所 以 单独 放 到 一 个 js 文件 中 ,方便 其 他 页 调用 ,js 文件 名 为 menujs。 调 用 时 在 页 面 中 加 入 如 下 代 
码 即 可 。 


<script src="JS/menu.js"></script> 
menujs 文件 中 显示 、 隐 藏 菜单 和 变换 背景 等 关键 函数 的 代码 如 下 : 
倒 程 15 代码 位 置 ， 资源 包 \TM\07\tmlogijs\menu.js 





上 * 浏 览 器 版 本 号 检测 */ 
var IE4=document.all&&navigator.userAgent.indexOf("Opera")==- 
var netscape6=document.getElementByld&&ldocument.all 
var netscape4=document.layers 
/* 显 示 下 拉 菜 单 函数 扩 
function showmenu(e,vmenu,mod){ 
/判断 浏览 器 版 本 六 
if (ldocument.all&&ldocument.getElementByld&&ldocument.layers) 
return 
A 菜单 内 容 */ 
which=vmenu 
调用 clearhidemenu() 函 数 */ 
clearhidemenu() 
让 调用 IE_clearshadow() 函 数 */ 
IE_clearshadow() 
/#* 根 据 不 同 的 浏览 器 ， 声 明 弹 出 式 菜单 小 
menuobj=IE4? document.all.popmenu : netscape6? document.getElementByld("popmenu") : netscape4? 
document.popmenu : ™ 
menuobij.thestyle=(IE4|Inetscape6)? menuobj.style : menuobj 
if (IE4|Inetscape6) 
/* 如 果 是 IE 4 或 Netscape 6 浏览 器 ， 就 调用 innerHTML 方法 输出 下 拉 菜 单 */ 
menuobj.innerHTML=which 
else{ 
/否则 使 用 write() 方 法 输出 下 拉 莱 单 */ 
menuobj.document.write(<layer name="other" bgColor="#E6E6E6" width="165" 
onmouseover="clearhidemenu()" onmouseout="hidemenu()">'+which+'</layer>') 


/关闭 数据 流 沁 
@ 
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menuobj.document.close() 
} 
户 调 用 不 同 的 方法 属性 ， 来 设置 不 同 浏览 器 的 宽 、 高 及 内 部 表格 的 大 小 、 位 置 */ 
menuobj.contentwidth=(IE4|Inetscape6)? menuobj.offsetWidth : menuobj.document.other.document.width 
menuobj.contentheight=(IE4|Inetscape6)? menuobj.offsetHeight : menuobj.document.other.document. height 
eventX=IE4? event.clientX : netscape6? e.clientX : e.x 
eventY=IE4? event.clientY : netscape6? e.clientY : e.y 
var rightedge=IE4? document.body.clientWidth-eventX : window.innerWidth-eventX 
var bottomedge=IE4? document.body.clientHeight-eventY : window.innerHeight-eventY 
if (rightedge<menuobj.contentwidth) 
menuobj.thestyle.left=IE4? document.body.scrollLeftreventX-menuobj.contentwidth+menuOffX : 
netscape6? window.pageXOffset+eventX-menuobj.contentwidth : eventX-menuobj.contentwidth 
else 
menuobij.thestyle.left=IE4? IE_x(event.srcElement}+rmenuOffX : netscape6? window.pageXOffset+reventX : 
eventX 
if (bottomedge<menuobj.contentheight&&mod!=0) 
menuobij.thestyle.top=IE4? document.body.scrollTop+eventY-menuobj.contentheight-event. offsetY+ 
menuOffY-23 : netscape6? window.pageY Offset+eventY-menuobj.contentheight-10 : eventY-menuobj.contentheight 
else 
menuobij.thestyle .top=IE4? IE_y(event.srcElement)+menuOffY : netscape6? window.pageYOffset+ 
eventY+10 : eventY 
人 * 设 置 标签 样式 为 可 见 */ 
menuobj.thestyle.visibility="visible" 
广 调 用 IE_dropshadow() 方 法 */ 
IE_dropshadow(menuobj,"#999999",3) 


return false 
| 
/隐藏 下 拉 菜 单 函数 沁 
function hidemenu(){ 
和 (windowmenuobj) 
menuobj.thestyle.visibility=(IE4|Inetscape6)? "hidden" : "hide" 
IE_clearshadow() 
} 


function dynamichide(e){ 
和 (IE4&&Imenuobj.contains(e.toElement)) 
hidemenu() 
else if (netscape6&&e.currentTargetl= erelatedTarget&& !contains_netscape6(e.currentTarget e.relatedTarget)) 
hidemenu() 


} 
A* 延 迟 隐藏 下 拉 莱 单 函 数 */ 
function delayhidemenu(X{ 
if (IE4|Inetscape6|Inetscape4) 
delayhide=setTimeout("hidemenu()",500) 


} 
/停止 隐藏 菜单 函数 


function clearhidemenu(){ 
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if (window.delayhide) 
clearTimeout(delayhide) 
站 


* 设 置 菜单 背景 */ 

人 * 鼠 标 移入 该 区 域 时 显示 的 背景 / 

function overbg(tdbgX{ 

tdbg.style.background='url( images/item_over.gif)' 

tdbg.style.border=' #9CA6C6 1px solid' 

} 

上 鼠标 移出 该 区 域 时 显示 的 背景 */ 

function outbg(tdbg}{ 

tdbg.style.background='url( images/item_out.gif)' 

tdbg.style.border=" 

} 

A 下拉 菜 单 内 容 */ 

A* 文 章 管 理 下 拉 菜 单 */ 

var productmenu='<table width=90><tr><td id=fileadd onMouseOver=overbg(fileadd) onMouseOut=outbg (fleadd)> 
<a href=file.php> 添 加 博客 文章 </a></td></tr>\ 

<tr><td id=query onMouseOver=overbg(query) onMouseOut=outbg(query)><a href=query.php> 查 询 博 客 文章 
</a></td></tr>\ 

<tr><td id=myfiles onMouseOver=overbg(myfiles) onMouseOut=outbg(myfiles)><a href=myfiles.php> 我 的 文章 
</a></td></tr></table>"' 

A* 图 片 管理 下 拉 菜 单 */ 

var Honourmenu='<table width=90><tr><td id=picadd onMouseOver=overbg(picadd) onMouseOut= outbg 
(picadd)><a href=add_pic.php> 添 加 图 片 </a></td></tr>\ 

<tr><td id=browse onMouseOver=overbg(browse) onMouseOut=outbg(browse)><a href=browse_pic.php> 浏 览 
片 </la></td></tr>\ 

<tr><td id=querypic onMouseOver=overbg(querypic) onMouseOut=outbg(querypic)><a href=query_pic.php> 查 
询 图 片 </a></td></tr></table>' 

/朋友 圈 管 理 下拉 菜 单刀 

var myfriend='<table width=90><tr><td id=friendadd onMouseOver=overbg(friendadd) onMouseOut=outbg (friendadd)> 
<a href=friend.php> 添 加 到 朋友 圈 </a></td></tr>\ 

<tr><td id=browse_fri onMouseOver=overbg(browse_fri) onMouseOut=outbg(browse_fri)><a href=browse_ fri.php> 
浏览 我 的 朋友 </a></td></tr>\ 

<tr><td id=cxfriend onMouseOver=overbg(cxfriend) onMouseOut=outbg(cxfriend)><a href=query_friend.php> 查 
询 朋 友信 息 </a></td></tr></table>" 

/管理 员 管 理 下 拉 菜 单刀 

var myuser='<table width=90><tr><td id=queryuser onMouseOver=overbg(queryuser) onMouseOut=outbg(queryuser)> 
<a href=queryuser.php> 查 询 用 户 信息 </a></td></tr>\ 

<tr><td id=browseuser onMouseOver=overbg(browseuser) onMouseOut=outbg(browseuser)><a href=browseuser. 
php> 浏 览 用 户 信 息 </a></td></tr>\ 

<tr><td id=managepub onMouseOver=overbg(managepub) onMouseOut=outbg(managepub)><a href=managepub. 
php> 公告 管理 </a></td></tr>\ 

</table>" 


& 
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7.10 ”登录 验证 码 技术 专题 


当今 网 络 安全 越 来 越 受到 人 们 的 重视 , 开发 一 个 带 验 证 码 的 用 户 登 录 模块 可 以 提高 网 站 的 安全 性 。 
其 中 ， 验 证 码 是 随机 生成 的 ， 可 以 是 字母 、 数 字 或 汉字 ， 也 可 以 是 图 片 。 本 节 技 术 专题 主要 就 是 学 习 
一 下 验证 码 的 几 种 实现 方法 及 技巧 。 


7.10.1 简单 的 数字 验证 


先 来 看 一 个 简单 的 数字 验证 码 : 生成 一 组 4 位 的 随机 数字 ， 每 刷新 一 次 ， 显 示 的 结果 都 不 相同 。 
如 果 验 证 码 输入 错误 ， 将 弹出 提示 窗口 。 数 字 验 证 码 运行 结果 如 图 7.26 所 示 。 


| 
EE 


localhost 明示 
到 不 这 葵 入 的 区 不 正史 


[CE 





图 7.26 简单 的 数字 验证 运行 结果 


简单 的 数字 验证 码 在 开发 过 程 中 主要 应 用 以 下 两 个 函数 。 
(1) mt rand0 函 数 。 
mt_rand() 函 数 主要 用 于 从 指定 参数 中 随机 取 一 个 数值 。 语 法 如 下 : 


int mt_rand([int min], [int max]); 
例如 ， 从 52 一 79 之 间 取 一 个 随机 数值 。 代 码 如 下 : 
mt_rand(52,79) 


(2) intval0 函 数 。 
intval0 函 数 主要 用 于 将 变量 转 成 整数 类 型 。 语 法 如 下 : 


int intval(mixed var, int [base]); 


intval0 函 数 可 将 变量 转 成 整数 类 型 。 可 省 略 的 参数 base 是 转换 的 基数 ， 默 认 值 为 10。 转 换 的 变量 
var 可 以 为 数组 或 类 之 外 的 任何 类 型 变量 。 


7.10.2 ”数字 图 形 验证 码 


数字 图 形 验证 码 相 对 于 数字 验证 码 而 言 ， 要 稍 复杂 一 些 。 数 字 图 形 验证 码 主要 应 用 mt_randO 函 数 


® 
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来 初始 化 一 组 4 位 的 随机 数 , 然后 利用 for 循环 语句 将 随机 生成 的 4 位 随机 验证 码 利用 数字 图 形 进 行 输 
出 。 图 形 验证 码 的 运行 结果 如 图 7.27 所 示 。 
到 于 国 尼 蛤 正 可 
RE 
于 至 | 
图 727 数字 图 形 验证 码 
实现 数字 图 形 验证 码 的 关键 代码 如 下 : 


<l--js 脚本 ， 判 断 文 本 框 中 输入 的 验证 码 是 否 匹 配 --> 
<script language="javascript"> 
function check(form}{ 
if(form.txt_yan.value=="™"){ 
alert(" 请 输入 验证 码 ");form .txt_yan.focus();return false; 

















} 
if(form.txt_yan.value!=form.txt_hyan.value}{ 
alert(" 对 不 起 ， 您 输入 的 验证 码 不 正确 "");form.txt_yan.focus();return false; 
} 
} 


</script> 


< 上 -验证 表单 -> 
<form name="form" method="post" action=""> 
<!-- 验 证 码 文本 框 --> 
<input type="text" name="txt_yan"> 
<!-- 生 成 随机 数字 图 形 验证 码 --> 
<?php 
<!- 生 成 随机 4 位 数 --> 
Snum=intval(mt_rand(1000,9999)); 
<!-- 使 用 for 循环 ， 输 出 随机 4 位 数 的 图 片 格式 --> 
for($i=0;,$i<4;,$i++X{ 
echo "<img src=images/checkcode/".substr(strval($num),$i,1).".gif>"; 
}?> 
<!- 将 随机 数 存 入 隐藏 域 --> 
<input type="hidden" name="txt_hyan" id="txt_hyan" value="<?php echo $num;?>"> 
<br> <br> 
<input type="submit" name="Submit" value=" 验 证 " onClick="return check(form);"> &nbsp; <input type="reset" 
name="Submit2" value=" 重 置 "> 
<!-- 处 理 代码 段 --> 
<?php 
/判断 验证 码 录 入 框 是 否 为 空 */ 
iflisset($_POSTT[txt_yan'])X{ 
上 * 当 输入 的 验证 码 和 隐藏 域 中 的 验证 码 相等 时 ， 输 出 欢迎 信息 */ 


if($_POST[txt_yan'] == $_POST[txt_hyan])f 
@ 
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echo "您 输入 的 验证 码 通过 ， 感 谢 您 的 加 盟 …"; 
» 


?> 


7.10.3 ”汉字 图 形 验证 码 


汉字 图 形 验证 码 是 目前 网 络 上 比较 流行 的 验证 码 方式 ， 这 种 验证 码 的 实现 有 一 定 的 难度 。 首 先 将 
图 片 所 对 应 的 汉字 依次 存储 到 数组 中 。 然 后 利用 Rand0 函 数 生成 一 个 小 于 10 的 数 ， 用 于 得 到 随机 字符 
的 位 置 ， 进 而 得 出 随机 字符 ， 并 设置 要 显示 的 随机 图 片 。 由 于 产生 的 随机 字符 串 是 4 位 的 ， 所 以 需要 
使 用 for 循环 语句 ， 根 据 数组 得 出 相应 图 片 的 路 径 ， 再 将 图 片 转换 成 数组 中 的 文字 ， 最 后 输出 到 浏览 
器 中 。 

以 汉字 图 形 化 显示 验证 码 的 完整 代码 如 下 : 


<?php 


$str=array(* 大 " "更 "" 创 "" 天 "" 科 "" 客 "" 情 "" 技 "" 立 "" 新 中 
$word=count($str); 

$img="; /初始化 变量 
$pic=""; // 初 始 化 变量 


for($i=0;,$i<4,$i++X{ 
S$num=rand(0,$word-1); 
Simg=$img."<img src=' images/checkcode/".$num.".gif width='16' height="16'>"; // 显 示 随 机 图 片 
S$pic=$pic.$str[$num]; /将 图 片 转换 成 数组 中 的 文字 





711 未 阐 总 站 


本 章 的 博客 管理 系统 首先 介绍 了 博客 的 基本 概念 、 发 展 前 景 、 影 响 范围 及 博客 网 的 功能 分 类 ， 使 
读者 对 当今 主流 博客 有 了 一 个 大 致 的 认识 。 其 次 ， 实 现 了 一 个 博客 系统 ， 其 中 包含 所 有 基本 功能 的 项 
目 开发 ， 使 读者 对 如 何 开发 一 个 博客 网 站 有 了 一 个 初步 的 了 解 。 最 后 ， 希 望 读者 通过 自己 的 努力 ， 来 
逐步 完善 和 加 强 这 个 博客 网 的 实用 功能 ， 最 终 达 到 一 个 令 自己 满意 的 作品 。 
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四 


全 兰 
恒 


365 影视 音乐 网 


(Apache+PHP+phpMyAdmin+MYySQL 5.5 实现 ) 


当今 社会 是 一 个 信息 快速 发 展 的 社会 ， 在 网 络 世界 浏览 信息 的 同 
时 也 可 以 听 听 歌曲 ， 这 样 既 恰 悦 了 身心 ， 又 获得 了 最 新 的 市 场 动 态 
由 此 网 络 上 出 现 了 很 多 的 影视 网 站 ， 很 爱 欢 迎 。 视 听 生 活 的 新 空间 ， 
必然 在 宽带 互联 网 上 开启 。“VOD (视频 点 播 )” 的 概念 已 经 被 越 来 越 
多 的 人 所 接受 ， 逐 渐 成 为 网 络 发 展 的 必然 趋势 之 一 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

WI 网 站 的 整体 设计 思路 及 实现 方法 

WI 应 用 <audio> 和 <video> 播 放 影音 文件 

mH ”PDO 的 使 用 

Mi 关闭 子 窗口 时 自动 刷新 父 窗口 

Hi 在 JavaScript 中 嵌入 PHP 代码 

让 文件 的 上 传 和 下 载 技术 
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8.1 开发 背景 














计算 机 技术 、 网 络 通信 技术 、 多 媒体 技术 及 数据 存储 技术 的 飞速 发 展 ， 对 人 类 的 
生产 和 生活 方式 产生 了 极 大 影响 。 网 络 以 其 特有 的 快速 、 高 效 、 便 捷 的 传输 方式 被 人 们 所 接受 。 随 着 
多 媒体 数据 的 存储 、 传 输 和 应 用 技术 的 不 断 成 熟 以 及 宽带 网 络 的 不 断 发 展 ， 我 们 相信 在 线 影视 音乐 点 
播 一 定 会 成 为 网 络 内 容 创新 的 重头 戏 。 


8.2 需求 分 析 


根据 客户 要 求 及 成 本 核算 ， 本 系统 所 要 实现 的 功能 如 下 : 




































































回 ”网 站 实行 会 员 等 级 管理 ， 分 “普通 会 员 ” 和 “高 级 会 员 ” 两 种 。 

回 ”网 站 提供 文件 上 传 、 下 载 和 在 线 视听 功 能 ， 不 同 的 会 员 等 级 实现 的 功能 操作 也 不 同 。 

回 ”网 站 分 前 台 和 后 台 两 部 分 ， 没 有 权限 的 用 户 无 法 进入 后 台 进行 管理 。 

回 ”独特 的 点 歌 模块 ， 为 网 站 的 会 员 提供 点 歌 平台 。 

回 ”前 期 数据 量 比较 少 ， 可 以 使 用 小 型 数据 库 以 节约 成 本 ， 如 果 后 期 数据 量 增 大 ， 则 可 以 更 换 大 

型 数据 库 。 
8.3 系统 分 析 

8.3.1 系统 功能 结构 

根据 365 影视 音乐 网 的 特点 ， 可 以 将 其 [365 路 视 音 乐风 六合 
分 为 前 台 和 后 台 两 个 部 分 设计 。 前 台 主 要 实 
现在 线 视 听 、 影 视 音乐 上 传 、 影 视 音乐 下 载 [ 在 网 条 听 ) (信息 展示 ] | (下载 排行 ] | ( 站 长 登录 ] 
和 在 线 点 歌 等 功能 ， 后 台 主 要 用 于 管理 员 。 「 | 思 诡 有 
对 影视 音乐 目录 、 数 据 信息 和 上 传 日 志 进 行 国 全 人 
管理 。 凤 | 加 标 | 到 得 

365 影视 音乐 网 的 前 台 功 能 结构 如 图 8.1 上 传 专区 】 (详细 信息 】 [( 人 ] 【会 员 专区 】 
所 示 。 


365 影视 音乐 网 的 后 台 功能 结构 如 图 8.2 


所 示 。 





如 赤水 蝇 


部 只 让 











间 
频 
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下 
共 








图 8.1 365 影视 音乐 网 前 台 功 能 结构 图 
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365 影 视 音乐 网 后 台 

































































(目录 管理 ] ( 数据 管理 】 ( 会 员 管理 ] (上 传 日 志 管 理 ] (管理 员 管理 ] 
视 | | 音 视 | | 音 | | 会 上 | | : 管 | | 管 | | 年 
频 | | 频 频 | | 频 员 | | 会 | | 会 传 | | 传 理 || 理 | | 理 
目 | | 日 数 | | 数 员 | | 员 | | 是 | | 晶 如 | | 但 | | 员 
录 | | 录 据 | | 据 | | 息 | | 涝 | | 型 | | 志 | | 志 洪 || 洪 | | 弄 
管 | | 管 管 | | 管 | | 查 | | 结 | | 除 | | 查 | | 汝 | | 罚 上 项 | 除 
理 | | 理 理 | | 理 | | 询 淘 | | 览 局 | | 昭 





图 8.2 365 影视 音乐 网 后 台 功能 结构 图 


8.3.2 功能 预览 


365 影视 音乐 网 由 多 个 功能 模块 组 成 , 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ,下面 列 出 几 个 
典型 功能 的 页 面 ， 其 他 页 面 参见 资源 包 中 的 源 程序 。 
影视 专区 页 面 如 图 8.3 所 示 ， 该 页 面 展 示 所 有 的 影片 信息 及 其 分 类 。 





图 8.3 影视 专区 (资源 包 \TM\08\online\list.php?action=video》 


音乐 专区 页 面 如 图 8.4 所 示 ， 该 页 面 展示 所 有 的 歌曲 信息 及 其 分 类 。 
影片 详情 页 面 如 图 8.5 所 示 ， 该 页 面 用 于 显示 影片 的 详细 信息 ， 包 括 影片 的 导演 、 主 演 、 类 型 以 及 
影片 简介 等 。 


@ 
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按 类 型 。 ” 汉 和 向 民 二 汪 车 后 :本寺 按 语言 “中 六 
沿 门 歌曲 
歌曲 分 类 歌曲 名 称 主 蝎 歌曲 简介 在 绪 试听 下 载 点 歌 
【多 村 襄 志 】 jhe 二 
【二 志 ] 子 和 突击 乱 回 素 Various Amists 
【村 网 民 如 ] 于/ 作怪 本 之 尖 
[时 机 原声] 地 寻 
[ 权 囊 民 王 ) 下 但 
【各 二 所 大 a EE 











图 8.4 音乐 专区 (资源 包 \TM\08\online\list.php?action=audio》 

















图 8.5 影片 详情 页 面 (资源 包 \TM\08\online\operation.php) 
查询 页 面 如 图 8.6 所 示 ， 


在 搜索 文本 框 中 输入 要 查询 的 影片 名 称 或 歌曲 名 称 ， 单 击 “ 搜 索 ” 按 钮 后 
即 可 显示 查询 结果 。 





B65 全 影视 音乐 网 [sess "as 








F365 影视 癌 世 Onlinc， 和 您 共享 终 由 

















图 8.6 查询 页 面 ( 资 源 包 \TM\08\online\show.php) 
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8.3.3 系统 流程 图 
365 影视 音乐 网 的 系统 流程 图 如 图 8.7 所 示 。 



































数据 库 检索 





























一 | 会 员 管理 | 
| 一 >| 信息 管理 | | 
一 ,| 上 传记 录 查 看 


图 8.7 365 影视 音乐 网 的 系统 流程 图 

















8.3.4 开发 环境 


在 开发 365 影视 音乐 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows 7。 

服务 器 : Apache 2.4.18。 

PHP 软件 : PHP 7.0.12。 

数据 库 : MySQL 5.5.47。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-3.5.8。 
开发 工具 : PhpStorm 2016.3。 

浏览 器 :Google Chrome。 

分 辩 率 : 最 佳 效果 为 1680x1050 像素 。 


2. 客户 端 


回 ”浏览 器 : 推荐 使 用 Google Chrome。 
回 ”分辩 率 : 最 佳 效果 为 1680x1050 像素 。 


8.3.5 文件 夹 组 织 结构 


365 影视 音乐 网 包括 前 台 和 后 台 两 部 分 ， 所 以 文件 夹 结构 也 主要 由 两 部 分 组 成 。365 影视 音乐 网 的 
组 织 结构 图 如 图 8.8 所 示 。 


@ 


办 办 办 办 
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7 Donline 一 一 一 一 一 一 一 一 网 站 根 目录 
v Dadmin 后 台 管 理 系统 目录 
上 户 comn 后 台数 据 库 连 接 文件 目录 
» Dess 后 台 CSS 样式 文件 目录 
» Dimages 后 台 图 片 文件 目录 
上 Dinc 后 台 自 定义 函数 及 公共 文件 目录 
* Djs 后 台 JavaScript 文件 目录 
» DD com 前 台数 据 库 连 接 文件 目录 
* Dess 前 台 CSS 样式 文件 目录 
> Ddata 网 站 数据 库 目录 
» Dimages 前 台 图 片 文件 目录 
» Oinc 前 台 自 定义 函数 及 公共 文件 目录 
» Dj 前 台 JavaScript 文件 目录 
» Dupfles 上 传 文件 管理 目录 








图 8.8 365 影视 音乐 网 的 文件 夹 组 织 结构 图 


8.4 ”数据 库 设计 





视频 讲解 
本 系统 使 用 的 是 MySQL 数据 库 ， 但 使 用 的 是 PDO 连接 方式 ， 这 是 为 了 便于 以 后 


数据 负担 加 重 后 ， 可 以 更 改 其 他 数据 库 ， 如 使 用 Oracle。 只 要 更 改 数据 库 而 无 须 重 新 编写 源 程序 。365 
影视 音乐 网 的 数据 库 db_online 中 共 包 含 8 个 数据 表 ， 这 8 个 数据 表 如 表 8.1 所 示 。 


表 8.1 数据 库 db_online 中 的 数据 表 


tb account 会 员 信息 表 
tb audio 音频 信息 表 
由 audiolist 音频 类 别 表 
由 gmade 等 级 限定 表 


说 了 明 


th_manager 管理 员 表 

th register 点 殉 信 息 表 
tb video 视频 信息 表 
th videolist 视频 类 别 表 


8.4.1 数据 库 概 念 设计 


通过 需求 分 析 和 功能 设计 ， 本 系统 规划 出 管理 员 信息 实体 、 会 员 信息 实体 、 视 频 信 息 实体 、 音 频 
信息 实体 、 视 频 目录 实体 和 音频 目录 实体 。 下 面 给 出 主要 的 实体 及 E-R 图 。 

会 员 信息 实体 包括 注册 用 户 的 详细 个 人 信息 ， 如 果 想 下 载 或 在 线 视听 ， 则 必须 注册 为 会 员 才 可 以 。 
会 员 信息 实体 E-R 图 如 图 8.9 所 示 。 

视频 文件 实体 包括 视频 名 称 、 视 频 图 片 、 视 频 文 件 、 主 要 演员 、 导 演 及 发 行商 等 多 项 资料 ， 视 频 
信息 实体 E-R 图 如 图 8.10 所 示 。 

音频 目录 实体 包括 音频 目录 分 类 的 相关 信息 ， 如 目录 名 称 、 目 录 级 别 、 父 目录 名 称 等 ， 音 频 目录 


多 
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实体 E-R 图 如 图 8.11 所 示 。 


地 址 这 注册。 《CS 次 


7 


目录 级 别 
有 4 - X 目 录 
(语言 称 
视频 文件 人 -也 音频 日 录 
二 Ss 
二 视频 类 鸡 二 1 (人 用 六 
Sa (nn) (Ra a CH 


图 89 会 员 信息 实体 ER 图 图 8.10 视频 信息 实体 ER 图 。 ”图 8.11 音频 目录 实体 E-R 图 
8.4.2 ”数据库 物理 结构 设计 


下 面 来 看 一 下 会 员 信息 表 、 音 频 信息 表 和 音频 类 别 表 的 结构 设计 。 
1. tb_account (会 员 信息 表 ) 
会 员 信息 表 主 要 存储 用 户 的 个 人 信息 ，tb_account 表 的 结构 如 表 8.2 所 示 。 


表 8.2 会 员 信息 表 结构 


Pp 






/ 









































名 字 | 类 型 | 整理 | 空 | 默 认 | 额 外 | 说 明 
id img) | 否 | 无 |Auro INCREMENT | 自动 篇 
name varchar(30) | utf8 general ci | 否 | 到 | | 会 员 昵称 
password varchar(30) | utf8 general ci | 否 | 玉 | | 会 员 密 码 
question varchar(50) | utf8 general ci | 天 | 到 [| | 密码 提示 问题 
answer varchar(50) | utf8 general ci | 否 | 到 | | 密码 提示 答案 
SeX varchar(10) | utf8 general ci | 否 | 天 | | 会 员 性 别 
age id | 再 | 无 | | 会 只 生 上 
job varchar(50) | utfg_general ci | 否 | 到 | | 会 员 职 业 
email varchar(100) | utfg_general ci 否 无 email 地 址 
address varchar(100) | utf§ general ci 否 无 联系 地 址 

qq varchar(20) | utf8§ general ci 否 无 联系 qq 
counts int(4) 否 0 上 传 次 数 
grade varchar(10) | utfg_general ci 否 普通 会 员 会 员 级 别 
whether varchar(10) | utf8 general ci 否 1 是 否 激活 
head varchar(200) | utf8 general ci 否 无 会 员 头 像 








2. tb_audio (音频 信息 表 )》 


音频 信息 表 主 要 用 来 存储 上 传 音频 的 资料 ， 如 音频 名 称 、 音 频 图 片 等 。tb_audio 表 的 结构 如 表 8.3 
所 示 。 


局 
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表 8.3 ”音频 信息 表 结 构 









































名 字 类 型 整理 空 默 认 额 外 说 了 明 
id int(4) 否 无 AUTO INCREMENT 自动 编号 
name varchar(100) | utf8 general ci 否 无 音频 名 称 
picture varchar(200) | utf8 general ci 否 无 封面 图 片 
actor varchar(100) | utf8 general ci 否 无 主唱 
ci varchar(50) | utf8 general ci 否 无 词 作者 
qu varchar(50) | utfg_general ci 否 无 曲 作者 
actortype varchar(50) | utf8 general ci 否 无 歌手 类 型 
style varchar(50) | utf8 general ci 否 无 歌曲 子 类 型 
publisher varchar(100) | utf8 general ci 否 无 发 行商 
froms varchar(100) | utf8 general ci 否 无 发 行 地 区 
type varchar(50) | utf8 general ci | 否 | 到 [| | 歌曲 父 类 型 
sizes varchar(50) | utf8 general ci | 否 | 天 | | 歌曲 大 小 
languages varchar(20) | utf8 general ci 否 | 无 | | 歌曲 语言 
publishTime | date | 否 | | | 发 行 时 间 
remark varchar(1000) | utf8_general ci | 否 | 无 [| | 歌曲 备注 
property varchar(20) | utfg_general ci | 否 | 到 [| | 歌曲 属性 
address varchar(200) | utf8 general ci | 否 | 到 [| | 歌曲 地 址 
UserName varchar(50) | utfg_general ci | 否 | 到 [| | 发 布 人 
issueDate datetime | | 发 布 时 间 
downTime | it | 百 | 无 | | 下 次数 
bool varchar(20) | utfg general ci | 否 | 1 | | 是 否 为 新 品 


3. tb_audiolist (音频 类 别 表 ) 


音频 类 刚 


表 主 要 是 存储 上 传 音频 所 属 的 类 型 〈 二 级 目录 ) 及 类 别 〈 一 级 目录 ) 。tb_audiolist 表 的 


结构 如 表 8.4 所 示 。 





表 8.4 音频 类 别 表 结 构 































































类 型 空 默 认 额 外 说 阴 
这 int(4) | 否 无 ”|AUTO INCREMENT | ”自动 编号 
grade varchar(50) | utf8_general_ci 否 无 | 类 型 级 别 
name varchar(50) | utf8_general ci 否 无 | 音频 子 类 型 
father varchar(50) | utf8_general_ci 否 无 | 音频 父 类 型 
userName varchar(50) | utf8 general ci 否 无 | 创建 人 
issueDate 否 无 创建 时 间 








由 于 篇 幅 有 限 ， 其 他 几 个 数据 表 的 结构 暂 不 介绍 。 
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视频 讲解 


8.5.1 前台 首页 概述 


365 影视 音乐 网 的 前 台 首 页 的 功能 模块 主要 包括 以 下 3 部 分 : 

回 ”网 站 头 部 : 主要 有 网 站 logo、 搜 索 模 块 、 注 册 / 登 录 模块 、 网 站 导航 及 电影 轮 播 图 。 

回 ”网 站 主 显示 区 : 包括 电影 信息 、 电 影 下 载 排行 榜 以 及 歌曲 信息 、 歌 曲 下 载 排行 榜 。 

回 ”网 站 底部 : 包括 版 权 信息 、 联 系 方式 等 。 

该 前 台 首 页 在 本 书 资源 包 中 的 路 径 为 “\TM\08\online\index.php”， 运 行 结果 如 图 8.12 所 示 。 


365 合影 机 音乐 网 
































we ee more tener ers eee ar 网 站 底部 


图 8.12 365 影视 音乐 网 前 台 首 页 
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8.5.2 前 台 首页 技术 分 析 


在 构建 前 台 首页 时 主要 使 用 了 引用 外 部 文件 的 方式 。 引 用 文件 是 指 将 另 一 个 源 文件 的 全 部 内 容 包 
含 到 当前 源 文件 中 进行 使 用 。 引 用 外 部 文件 可 以 减少 代码 的 重用 性 ， 是 PHP 编程 的 重要 技巧 。PHP 提 
供 了 include 语句 、require 语句 、include_once 语句 和 require_once 语句 用 于 实现 引用 文件 ， 在 该 网 站 
前 台 首页 中 使 用 的 是 include 语句 。 

使 用 include 语句 引用 外 部 文件 时 ， 只 有 代码 执行 到 include 语句 时 才 将 外 部 文件 引用 进来 并 读 取 
文件 的 内 容 ， 当 所 引用 的 外 部 文件 发 生 错误 时 ， 系 统 只 给 出 一 个 警告 ， 而 整个 php 文件 则 继续 向 下 执 
行 。include 语句 的 语法 如 下 : 


Void include(string filename); 


参数 filename 是 指定 的 完整 路 径 文件 名 。 
例如 ， 在 index.php 文件 中 引入 头 部 文件 top.php 的 代码 如 下 : 








<?php 


include("top.php"); 
?> 





8.5.3 ”前 台 首页 的 实现 过 程 


国 ”前 台 首页 使 用 的 数据 表 : tb_audio、tb_video 

在 首页 中 ， 使 用 到 了 4 个 include 语句 ,分 别 将 网 站 头 部 页 面 、 主 显示 页 面 、 排 行 榜 页 面 以 及 版 权 
信息 页 面 等 主要 的 模块 加 载 进来 。 限 于 篇 幅 ， 这 些 页 面 的 代码 不 做 详细 介绍 ， 具 体 代 码 请 参考 本 书 附 
带 资源 包 。 前 台 首 页 index.php 的 实现 代码 如 下 : 


倒 程 01 代码 位 置 : 资源 包 \TM\08\online\index.php 
<?php 
session_start(); /启动 session 
?> 
<script src="js/chk.js"></script> 
<script type="text/javascript"> 
function chk_see(){ 


var name = "<?php echo isset($_SESSION[name])?$_SESSION[name]:";?>":// 为 变量 赋值 
if(name == ™X{ // 如 果 变 量 name 的 值 为 空 

alert(' 只 有 会 员 才 能 观看 影片 ， 请 登录 ! '); // 弹 出 对 话 框 

return false; // 返 回 false， 用 户 无 法 观看 影片 
}else{ 

retum true; /返回 true， 会 员 可 以 观看 影片 


} 
} 
function chk_listen()\{ 
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var name = "<?php echo isset($_SESSION[name])?$_SESSION[name]:"";?>";/ 为 变量 赋值 


if(name == "YX{ /| 如果 变 量 name 的 值 为 空 
alert( 只 有 会 员 才能 播放 歌曲 ， 请 登录 ! '); /弹出 对 话 框 
return false; /返回 false， 用 户 无 法 播放 歌曲 

}else{ 
return true; /| 返回 true， 会 员 可 以 播放 歌曲 

} 

} 
</script> 


<link rel="stylesheet" href="css/style.css" /> 
<body onload="setlnterval(changeimage(),3000)"> 


<?php 
include("top.php"); /1 引入 logo、 导 航 页 面 
?> 
<table width="100%" border="0" cellpadding="0" cellspacing="0" bgcolor="##ffffff"> 
<tr> 
<td>&nbsp;</td> 
<td width="960" valign="top"> 
<?php 
include("main.php"); /引入 主 显示 页 面 
?> 
</td> 
<td width="240" valign="top"> 
<?php 
include("right.php"); 1/ 引 入 电影 榜 、 歌 曲 排行 榜 页 面 
?> 
</td> 
<td>&nbsp;</td> 
</tr> 
</table> 
<?php 
include("bottom.php"); 1/ 引 入 版 权 信息 、 联 系 方式 页 面 
?> 
</body> 





8.6 详细 信息 查看 模块 


8.6.1 详细 信息 查看 模块 概述 


详细 信息 查看 模块 主要 包括 视频 详细 信息 查看 和 音频 详细 信息 查看 两 个 部 分 。 查 看 信息 功能 没有 
权限 限制 ， 无 论 是 游客 还 是 会 员 都 可 以 查看 。 但 查看 页 面 中 的 播放 影片 或 播放 歌曲 的 功能 则 是 只 有 会 
员 登 录 后 才 可 以 实现 的 操作 ， 而 下 载 影片 或 下 载 歌 曲 的 功能 只 有 高 级 会 员 登 录 后 才能 实现 。 视 频 详细 
信息 页 面 的 运行 结果 如 图 8.13 所 示 。 


全 
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电 关 于 所 














图 8.13 视频 详细 信息 页 面 
8.6.2 ”详细 信息 查看 模块 技术 分 析 


在 这 个 模块 中 ， 所 要 解决 的 技术 难点 就 是 : 系统 需要 根据 不 同 的 浏览 用 户 ， 弹 出 不 同 的 操作 权限 
对 话 框 。 在 JavaScript 代码 中 使 用 让 语句 ， 并 配合 PHP 中 的 session 技术 来 实现 该 功能 。 以 视频 详细 信 
息 页 面 为 例 ， 首 先 通过 检测 $_SESSION[mame'] 是 否 存在 ， 来 判断 用 户 是 否 登录 ， 如 果 没 有 登录 ， 在 单 
击 “ 播 放 影片 按钮 时 会 弹出 “只 有 会 员 才能 观看 影片 , 请 登录 ! ”的 提示 , 然后 判断 $_SESSION['grades'] 
是 否 为 “高 级 会 员 ”， 如 果 用 户 未 登录 或 登录 会 员 为 普通 会 员 ， 在 单 击 “ 下 载 影 片 ” 按 钮 时 会 弹出 “只 
有 高 级 会 员 登 录 后 才能 下 载 ”的 提示 。 该 功能 的 关键 代码 如 下 : 


倒 程 02 ”代码 位 置 ， 资源 包 \TM\08\online\v_intro.php 


<script type="text/javascript"> 
function chk_see(){ 
var name = "<?php echo isset($_SESSION[name"])?$_SESSION[name"]:";?>";// 为 变量 赋值 





if(name == ™"}{ // 如 果 变 量 name 的 值 为 空 
alert(' 只 有 会 员 才 能 观看 影片 ， 请 登录 ! '); // 弹 出 对 话 框 
return false; // 返 回 false， 用 户 无 法 观看 影片 
Jelse{ 
return true; // 返 回 true， 会 员 可 以 观看 影片 


4 
function chk_download(X{ 
var name = "<?php echo isset($_SESSION[name'])?$_SESSION[name1]:";?>";// 获 取 用 户 名 
// 获 取 用 户 等 级 
var grades = "<?php echo isset($_SESSION[grades])?$_SESSION[grades]:";?>"; 
if(name == "" || grades!=" 高 级 会 员 "X{// 如 果 变 量 name 的 值 为 空 或 者 变量 grades 的 值 不 是 "高 级 会 员 " 


alert(' 只 有 高 级 会 员 登 录 后 才能 下 载 '); // 弹 出 对 话 框 

return false; // 返 回 false， 用 户 无 法 进行 下 载 
Jelse{ 

return true; // 返 回 true， 用 户 可 以 进行 下 载 


下 
} 
</script> 





_ 国 ) 
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8.6.3 详细 信息 查看 的 实现 过 程 


国 ”信息 查看 模块 使 用 的 数据 表 : tb audio、tb video 

用 户 可 以 在 视频 信息 展示 页 面 单 击 司 图 标 进 入 视频 详细 信息 页 面 。 在 视频 详细 信息 页 面 
(Vv_intro.php》 中， 以 浏览 器 地 址 栏 中 传递 的 id 值 为 条 件 对 tb_video 数据 表 进 行 查询 ， 并 将 查询 结果 
中 的 主要 数据 输出 到 页 面 中 ， 包 括 影 片 名 称 、 导 演 、 主 演 及 类 型 等 信息 。 关 键 代码 如 下 : 


倒 程 03 ”代码 位 置 ， 资源 包 \TM\08\online\_intro.php 


<?php 
$sql="select * from tb_video where id=".$_GETTid]; // 定 义 查询 语句 
$result = $pdo->prepare($sql); /准备 查询 
$result->execute(); /执行 查询 
if$rst=$result->fetch(PDO::FETCH_NUM)X // 将 查询 结果 返回 到 数组 并 判断 是 否 为 真 
?> 


<table width="100%" border="0" cellspacing="0" cellpadding="0" bgcolor="#ffffff" 
class="w_1200" style="margin-top: 20px;margin-bottom: 20px; padding-bottom:36px;"> 
<tr> 
<td width="34">&nbsp;</td> 
<td colspan="3"> 
<span style="font-size: 20px;color: #2bb673; border-bottom: 4px solid #2bb673; 
line-height: 54px; margin: 0 0 6px 0px; padding-bottom: 10px"> 电 影 详情 </span> 
</td> 
</tr> 
<tr><td width="34"></td><td colspan="2" height="1" bgcolor="#eSeSe5"></td></tr> 
<tr> 
<td></td> 
<td align="left" valign="top" style="padding-top:30px;"> 
<table width="95%" border="0" cellspacing="0" cellpadding="0" > 
<tr> 
<td width="30%" height="" align="|left" valign="middle"> 
<img name="news" src="<?php echo "upfiles/video/".$rst[2]; ?>" width="280" height="362" alt="" 
border="0" style=" border-color:#CCCCCC; margin-top:15px; margin-left:0px; 
margin-bottom:15px; margin-right:0px;" /> 
</td> 
<td width="70%" align="left" valign="top"> 
<table border="0" cellspacing="0" cellpadding="0" style="margin-top:10px;"> 
<tr> 
<td align="left" colspan="2" height="60" style="font-size:24px; 
font-weight:bolder;"><?php echo $rst[1]; ?></td> 
</tr> 
<tr> 
<td width="280" height="50" align="left" style="font-size:18px;"> 
导演 : <?php echo S$rst[7]; ?></td> 
</tr> 
<tr> 
<td height="50" align="left" style="font-size:18px;"> 
主演 : <?php echo Srst[6]; ?></div></td> 
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</tr> 
<tr> 
<td height="50" align="left" style="font-size:18px:;"> 
类 型 : <?php echo Srst[11]; ?></td> 
</tr> 
<tr> 
<td height="50" align="left" style="font-size:18px;"> 
语言 : <?php echo Srst[9]; ?></td> 
</tr> 
<tr> 
<td height="50" align="left" style="font-size:18px;"> 
发 行 时 间 : <?php echo Srst[13]; ?></td> 
</tr> 
</table> 
</td> 


<td height="58" align="left"> 
<a href="operation.php?action=see&id=<?php echo $rst[16]; ?>" 
target="_blank" onclick="return chk_see()"><img src="images/play.png"></a> 
<a href="download.php?id=<?php echo S$rst[16]; ?>&action=video" 
onClick="return chk_download()"><img src="images/download.png"></a> 
</td> 
</tr> 
<tr> 
<td height="48" colspan="2" style="font-size:18px;">&nbsp;&nbsp; 影 片 详情 :</td> 
</tr> 
<tr> 
<td colspan="2" style="font-size:14px; line-height:30px;">&nbsp;&nbsp; 
<?php echo $rst[23]; ?></td> 
</tr> 
</table> 
</td> 
</tr> 
</table> 
<?php 
} 


?> 


8.7 在线 观 看 与 试听 模块 设计 








视频 讲解 





8.7.1 在 线 观 看 与 试听 模块 概述 


用 户 登 录 后 可 以 在 线 观 看 视频 。 在 不 同 的 页 面 都 可 以 直接 进入 在 线 观 看 页 面 ， 包 括 首 页 、 视 频 列 
表 页 、 视 频 搜索 结果 页 和 视频 详细 信息 页 。 但 必须 是 登录 会 员 才 可 以 ， 游 客 是 没有 权限 的 。 会 员 可 以 
在 首页 或 视频 信息 展示 页 面 单 击 视频 图 片 或 视频 名 称 ， 在 视频 详细 信息 页 面 单 击 “播放 影片 ”按钮 来 


351 
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进行 在 线 观看 。 在 线 观 看 页 面 的 运行 结果 如 图 8.14 所 示 。 

用 户 登 录 后 还 可 以 在 线 试听 音频 数据 。 在 不 同 的 页 面 都 可 以 直接 进入 在 线 试听 页 面 ， 包 括 首页 、 
歌曲 列表 页 、 歌 曲 搜索 结果 页 和 音频 详细 信息 页 。 但 必须 是 登录 会 员 才 可 以 ， 游 客 是 没有 权限 的 。 会 
员 可 以 在 首页 单 击 歌曲 图 片 或 歌曲 名 称 ， 在 音频 信息 展示 页 面 单 击 > 图 标 ， 或 者 在 音频 详细 信息 页 面 单 
击 “ 播 放歌 曲 ” 按 钮 来 进行 在 线 试 听 。 在 线 试听 页 面 的 运行 结果 如 图 8.15 所 示 。 


11 010 1309 和 一 一 一 哆 一 人 





图 8.14 ”在线 观看 页 面 图 8.15 在 线 试听 页 面 
8.7.2 在 线 观 看 与 试听 模块 技术 分 析 


在 该 网 站 中 ,实现 影片 的 在 线 观看 功能 使 用 的 是 HIML5 中 新 增 的 video 元 素 ， 而 实现 歌曲 的 在 线 
试听 功能 使 用 的 是 HTMLS5 中 新 增 的 audio 元 素 。 下 面 对 这 两 个 元 素 进行 详细 介绍 。 
1. video 元 素 


video 元 素 专门 用 来 播放 网 络 上 的 视频 或 电影 ， 该 元 素 的 使 用 方法 很 简单 ， 只 要 设 定好 元 素 的 长 、 
宽 等 属性 ， 并 且 把 播放 视频 的 URL 地 址 指定 给 该 元 素 的 sre 属性 即 可 ，video 元 素 的 使 用 方法 如 下 ; 





<video width="500" height="300" src="movie.mp4" controls="controls" autoplay="autoplay "> 
您 的 浏览 器 不 支持 video 元 素 ! 
</video> 


video 元 素 的 属性 及 其 描述 如 表 8.5 所 示 。 
表 8.5 video 元 素 的 属性 及 其 描述 





述 










描述 属 性 描 
width | 设置 视频 播放 器 的 宽度 autoplay 设置 自动 播放 
height | 设置 视频 播放 器 的 高 度 loop 设置 循环 播放 
设置 视频 在 页 面 加 载 时 进行 加 载 , 并 预备 播 
放 。 如果 使 用 了 autoplay 属性 则 忽略 该 属性 











STC 要 播放 视频 的 URL preload 
添加 播放 、 暂 停 和 音量 控件 
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/ 
Dep 


另外 ， 还 可 以 通过 使 用 source 元 素来 为 同一 个 媒体 数据 指定 多 个 播放 格式 ， 以 确保 浏览 器 可 以 从 
中 选择 一 种 自己 支持 的 播放 格式 进行 播放 ， 浏 览 器 的 选择 顺序 为 代码 中 的 书写 顺序 ， 它 会 从 上 往 下 判 
断 自己 对 该 播放 格式 是 否 支持 ， 直 到 选择 到 自己 支持 的 播放 格式 为 止 。 其 使 用 方法 如 下 : 
<video width="500" height="300"> 
<!-- 在 ogg 格式 与 MP4 格式 之 间 选 择 自己 支持 的 播放 格式 --> 
<source src="demo/movie.0gg " type="Video/ogg"/> 


<source src="demo/movie.mp4" type="video/mp4"/> 
</video> 


source 元 素 中 的 src 属性 是 指 播放 媒体 的 URL 地 址 ，type 属性 表示 媒体 类 型 ， 其 属性 值 为 播放 文 
件 的 MIME 类 型 。 


2. audio 元 素 


audio 元 素 专门 用 来 播放 网 络 上 的 音频 数据 ,该 元 素 的 使 用 方法 也 很 简单 ， 只 要 把 播放 音频 的 URL 
地 址 指定 给 该 元 素 的 src 属性 即 可 ，audio 元 素 的 使 用 方法 如 下 : 





<audio src="song.mp3" controls="controls" autoplay="autoplay"> 
您 的 浏览 器 不 支持 audio 元 素 ! 
</audio> 


audio 元 素 的 属性 及 其 描述 如 表 8.6 所 示 。 
表 8.6 audio 元 素 的 属性 及 其 描述 








属 性 描述 
SIC 要 播放 音频 的 URL 

controls 添加 播放 、 和 暂停 和 音量 控件 

autoplay 设置 自动 播放 

loop 设置 循环 播放 





preload 设置 音频 在 页 面 加 载 时 进行 加 载 ， 并 预备 播放 。 如 果 使 用 了 autoplay 属性 则 忽略 该 属性 


人 
| 元 素 ， 则 会 显示 <audio> 与 <laudio> 之 间 设 置 的 内 容 。 


另外 , 在 audio 元 素 中 也 可 以 使 用 多 个 source 元 素 ，source 元 素 可 以 链接 不 同 的 音频 文件 ,浏览 器 
将 选择 第 一 个 支持 的 格式 。 其 使 用 方法 如 下 : 


<audio controls="controls" autoplay="autoplay"> 
<!-- 在 ogg 格式 与 MP3 格式 之 间 选 择 自己 支持 的 播放 格式 --> 
<source src="demo/song.0gg " type="audio/ogg"/> 
<source src="demo/song.mp3" type="audio/mpeg"/> 
</audio> 


四 
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source 元 素 中 的 src 属性 是 指 播放 媒体 的 URL 地 址 ，type 属性 表示 媒体 类 型 ， 其 属性 值 为 播放 文 
件 的 MIME 类 型 。 


8.7.3 ”在线 观看 的 实现 过 程 


国 。 在 线 视听 模块 使 用 的 数据 表 : tb_video 

在 视频 信息 展示 页 面 中 ， 用 户 单 击 罚 图 标 ， 打 开 视 频 播放 页 面 (see.php) 进行 在 线 观看 。 视 频 播 
放 页 面 主要 是 根据 传递 的 参数 进行 数据 库 检 索 ， 并 将 对 应 数据 的 硬盘 存储 地 址 作为 多 媒体 文件 的 引用 
地 址 ， 当 页 面 加 载 完 毕 后 ， 视 频数 据 将 自动 播放 ， 实 现在 线 观看 功能 。 在 线 观 看 页 面 的 主要 代码 如 下 : 


倒 程 04 ”代码 位 置 ， 资源 包 \TM\08\online\see.php 


<?php 

header ( "Content-type: text/html; charset=utf-8" ); /设置 文件 编码 格式 

session_start(); /启动 Session 

if(isset($_SESSION['name'])}{ /| 如果 用 户 已 登录 
?> 


<body leftmargin="0" topmargin="0"> 
<video src="upfiles/video/<?php echo $_GET['id"];?>" width="100%" height="100%" controls="controls" 
autoplay="autoplay"></video> 


</body> 
<?php 
Jelse{ 
echo "<script>alert(' 只 有 会 员 才能 观看 影片 ， 请 登录 ! '); 
location.href='login.php';</script>"; /弹出 提示 信息 





8.7.4 在 线 试听 的 实现 过 程 


国 ”在 线 视听 模块 使 用 的 数据 表 : tb_ audio 

在 音频 信息 展示 页 面 中 ， 用 户 单 击 P> 图 标 ， 打 开 音频 播放 页 面 (listen.php〉 进 行 在 线 试听 。 在 试听 
页 面 中 ， 根 据 传递 的 参数 进行 数据 库 检 索 ， 并 将 对 应 数据 的 硬盘 存储 地 址 作为 音频 文件 的 引用 地 址 ， 
当 页 面 加 载 完 毕 后 ， 音 频数 据 将 自动 播放 ， 实 现在 线 试听 的 功能 。 在 线 试 听 页 面 的 主要 代码 如 下 : 


倒 程 05 ”代码 位 置 : 资源 包 \TM\08\online\listen.php 


<?php 

session_start(); /启动 Session 
header("Content-type:text/html:charset=utf-8"); // 设 置 文件 编码 格式 
if(isset($_SESSION[name'])}{ /如 果 用 户 已 登录 


?> 
<body leftmargin="0" topmargin="0"> 
<audio src="upfiles/audio/<?php echo $_GET[id];?>" controls="controls" autoplay="autoplay"></audio> 
</body> 


© 
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<?php 
Jelse{ 
echo "<script>alert(' 只 有 会 员 才 能 播放 歌曲 ， 请 登录 ! '); 
top.location.href="|login.php';</script>"; ”// 弹 出 提示 信息 并 跳 转 页 面 





8.8 ”点 歌 模块 设计 





8.8.1 点 歌 模块 概述 


本 项 目的 在 线 点 歌 模块 实现 了 会 员 间 发 送 祝福 与 点 歌 的 功能 , 收 到 祝福 的 会 员 可 以 实现 在 线 听 歌 。 
用 户 通过 音频 信息 展示 页 面 进入 点 歌 功 能 模块 。 单 击 月 图 
标 ， 可 以 打开 点 歌 页 面 为 其 他 用 户 进行 点 歌 ， 点 歌 接收 人 点 歌 记 录 详 单 
在 登录 之 后 就 可 以 查看 点 歌 信 息 ， 对 歌曲 进行 试听 。 进 行 。 | 并 收入; 多 
点 歌 的 前 提 条 件 是 用 户 必 须 登录 。 点 歌 页 面 的 运行 结果 如 。。 | 全， 机 天! 
图 8.16 所 示 。 


8.8.2 ”点 歌 模块 技术 分 析 


用 户 在 进行 点 歌 时 会 将 点 歌 信 息 存储 到 数据 库 中 ， 点 
歌 接 收入 在 登录 之 后 可 以 查看 数据 库 中 存储 的 点 歌 信息 。 本 系统 使 用 了 PDO 方式 操作 数据 库 ， 在 执行 
点 歌 操作 时 使 用 了 PDO 中 的 prepare0、execute0、fetchO0 和 exec0 方 法 ， 下 面 分 别 进行 介绍 。 


1. prepare() 和 execute() 方 法 


prepare0 和 execute0 两 个 方法 是 PDO 中 的 两 个 预 处 理 语句 。 首 先 ， 通 过 prepare0 方 法 做 查询 的 准 
备 工作 ,然后 ,通过 execute() 方 法 执行 查询 .并 且 还 可 以 通过 bindParam() 方 法 来 绑 定 参数 提供 给 executeO 
方法 。 其 语法 如 下 : 





图 8.16 点 歌 页 面 





PDOStatement PDO::prepare(string statement [, array driver_options]) 
bool PDOStatement::execute([array input_parameters]) 


2. fetch() 方 法 
fetch0 方 法 用 于 获取 结果 集中 的 下 一 行 ， 其 语法 格式 如 下 : 
mixed PDOStatement::fetch([int fetch_style [, int cursor_orientation [, int cursor_offset]]]) 


参数 fetch_style 为 控制 结果 集 的 返回 方式 ， 其 可 选 方式 如 表 8.7 所 示 ; 参数 cursor_orientation 是 
PDOStatement 对 象 的 一 个 滚动 游标 ， 可 用 于 获取 指定 的 一 行 ， 参数 cursor_offset 为 游标 的 偏 移 量 。 


@ 
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表 8.7 fetch_style 控制 结果 集 的 可 选 值 
值 说 明 
PDO:FETCH ASSOC | 关联 数组 形式 
PDO::FETCH_NUM | 数字 索引 数组 形式 
PDO:FETCH_BOTH | 两 者 数组 形式 都 有 ， 这 是 默认 的 
| 
| 














PDO::FETCH OBJ 按照 对 象 的 形式 ， 类 似 于 以 前 的 mysqli_fetch_objectO0 
PDO::FETCH BOUND 以 布尔 值 的 形式 返回 结果 ， 同 时 将 获取 的 列 值 赋 给 bindParam() 方 法 中 指定 的 变量 


PDO::FETCH LAZY 以 关联 数组 、 数 字 索 引 数 组 和 对 象 3 种 形式 返回 结果 


3. exec() 方 法 
exec() 方 法 返回 执行 SQL 语句 后 受 影 响 的 行 数 ， 其 语法 如 下 : 











int PDO::exec(string statement) 





参数 statement 是 要 执行 的 SQL 语句 。 该 方法 返回 执行 SQL 语句 后 受 影响 的 行 数 ， 通 常用 于 
INSERT、DELETE 和 UPDATE 语句 中 。 


8.8.3 会 员 点 歌 的 实现 过 程 


国 。 点 歌 模块 使 用 的 数据 表 : tb_register、tb_audio 

单 击 导航 栏 中 的 “音乐 ” 超 链 接 ， 首 先进 入 到 歌曲 信息 页 面 ， 页 面 中 的 在 线 试听 和 详细 信息 查看 
在 其 他 模块 中 都 有 介绍 ， 这 里 主要 讲解 点 歌 功能 。 单 击 另 图标， 进入 点 歌 页 面 。 用 户 在 点 歌 页面 需 要 
输入 接收 人 用 户 名 和 祝福 语 。 在 点 歌 页 面 ， 将 上 一 页 传递 过 来 的 歌曲 id、 接 收入 及 祝福 语 等 信息 组 成 
insert 语句 存储 到 数据 库 中 ， 然 后 刷新 父 窗口 并 关闭 当前 页 面 。 关 键 代 码 如 下 : 


倒 程 06 ”代码 位 置 : 资源 包 \TM\08\online\give.php 





<?php 
session_start(); /启动 Session 
include "conn/conn.php"; // 包 含 数据 库 连接 文件 
ifisset($_POST[toname]) &&$_POST[toname'] <>""){ // 如 果 toname 参数 值 不 为 空 
$sql="select * from tb_account where name=".$_POST[toname1].”; // 定 义 查询 语句 
Sresult=$pdo->prepare($sql); // 准 备查 询 
Sresult->execute(); /执行 查询 
if(!$rst=$result->fetch(PDO::FETCH_NUM)X // 如 果 查 询 结果 不 为 真 
echo "<script>alert( 该 会 员 不 存在 ， 请 重新 输入 ! ');</script>";，”// 弹 出 提示 信息 
exit(); // 退 出 程序 
} 
$id=$_POSTTid"]; /将 歌曲 id 赋值 给 变量 
S$toname=$_POST[toname'; // 将 点 歌 接收 入 赋值 给 变量 
$from=$_SESSION[name']: /将 点 歌 人 赋值 给 变量 
$remark=$_POST[remark]; /将 点 歌 祝福 语 赋值 给 变量 
/定义 插入 语句 


$sql="insert into tb_register 
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values(",".$id.",".$from.",".$toname.",".$remark.",".date("Y-m-d H:i:s").")"; 
Srst = $pdo->exec($sql); /执行 语句 
?> 
<script language="javascript"> 
<?php 
if(!(Srst == false)){ 


?> 

alert(" 点 歌 信息 保存 成 功 "); // 弹 出 对 话 框 
<?php 

Jelse{ 


ys 
alert(" 点 歌 失败 "); // 弹 出 对 话 框 
<?php 


?> 
top.openerlocation.reload(); /刷新 父 窗口 
top.window .close(); /关闭 当前 页 面 
</script> 
<?php 


?> 





8.8.4 查看 点 歌 信息 的 实现 过 程 





在 点 歌 操作 完成 后 ， 点 歌 接 收 人 需要 登录 系统 ， 通 过 单 点 歌 信 息 查 看 
击 鼠 标 指 向 头像 时 弹出 的 “查看 点 歌 信息 ” 超 链接 来 进行 查 Er ee 
看 。 运 行 结果 如 图 8.17 所 示 。 机 后 : 隔 耐克 - 帮 作 天 天际 0 全 
由 图 8.17 可 以 看 出 ， 在 点 歌 信息 查看 页 面 中 显示 了 点 播 
的 歌曲 名 称 、 点 歌 人 以 及 收 到 的 祝福 语 ， 用 户 只 需要 单 击 歌 亚 旧 
曲名 称 就 可 以 进行 在 线 试听 , 试听 后 该 点 歌 信息 将 会 被 删除 。 图 8.17 查看 点 歌 信 息 


在 点 歌 信息 查看 页 面 s musicphp 中 , 首先 包含 数据 库 连 
接 文件 ， 然 后 查询 以 登录 用 户 作为 接收 人 的 点 歌 信息 ， 并 应 用 do…while 语句 将 点 播 的 歌曲 名 称 、 点 歌 
人 以 及 收 到 的 祝福 语 循环 输出 到 页 面 中 ， 关 键 代 码 如 下 : 


倒 程 07 ”代码 位 置 : 资源 包 \TM\08\online\s_music.php 


<?php 
include "conn/conn.php"; // 包 含 数 据 库 连 接 文件 
$s_sqlstr="select * from tb_register where toName=".$_SESSION[name]." order by 
issueDate Desc"'; /定义 查询 语句 
Sresult = $pdo->prepare($s_sqlstr); 儿 准 备查 询 
Sresult->execute(); /执行 查询 
访 $s_rst=$result->fetch(PDO::FETCH_NUM)X /将 查询 结果 返回 到 数组 中 并 判断 结果 是 否 为 真 
dof // 循 环 输出 查询 结果 
?> 


<table width="400" border="0" cellspacing="0" cellpadding="0"> 


@ 
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<?phi 


?> 


<tr valign="top"> 
<td height="15" colspan="4">&nbsp;</td> 
</tr> 
p 
$s_sqlstr1="select * from tb_audio where id=".$s_rst[1]; /定义 查询 语句 
$result1 = $pdo->prepare($s_sqlstr1); // 准 备查 询 
Sresult1->execute(); /执行 查询 
if($s_rst1 = $result1->fetch(PDO::FETCH_NUM)X{ // 将 查询 结果 返回 数组 并 判断 是 否 为 真 
<tr valign="top"> 


<td width="110" height="30"> 歌 曲名 称 : </td> 
<td width="185" height="30"><a href="operation.php?action=dotlisten&id= 


<?php echo $s_rst[0]; ?>&name=<?php echo $s_rst1[16]; ?>"><?php echo $s_rst1[1]; ?></a></td> 


<?p| 


jel 





<td width="75" height="30"> 点 歌 人 : </td> 
<td width="103" height="30"> 
<?php echo $s_rst[2]; ?></td> 
</tr> 
<tr valign="top"> 
<td width="110"> 祝 语 : </td> 
<td height="55" colspan="3"> 
<textarea name="textarea" cols="40" rows="3" > 
<?php echo $s_rst[4]; ?></textarea> 
</td> 
</tr> 
</table> 
hp 
} 
}while($s_rst=$result->fetch(PDO::FETCH_NUM)); 


se{ 
echo " 暂 无 点 歌 信息 ! "; 





根据 用 户 对 各 个 功能 模块 的 使 用 频率 和 重要 程度 ， 本 系统 后 台 首 页 中 要 显示 的 模块 主要 有 以 下 两 


部 分 : 


(1) 网 站 左 侧 导 航 栏 ， 包 括 各 个 管理 模块 及 分 类 。 


目录 管理 模块 : 主要 包括 视频 目录 管理 和 音频 目录 管理 两 个 部 分 。 
数据 管理 模块 : 主要 包括 视频 数据 管理 和 音频 数据 管理 两 个 部 分 。 
会 员 管理 模块 : 主要 包括 会 员 等 级 设置 和 会 员 数 据 管理 两 个 部 分 。 
上 传 信息 管理 模块 : 主要 用 于 浏览 和 查询 用 户 上 传 操作 的 详细 内 容 。 
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回 ”管理 员 设 置 模块 : 主要 包括 管理 员 信息 添加 、 删 除 及 冻结 3 个 部 分 。 
(2) 网 站 主 显示 区 : 显示 各 个 模块 的 操作 及 结果 。 
该 后 台 首 页 在 本 书 资源 包 中 的 路 径 为 “\TM\08\online\admin\main php”， 运 行 结果 如 图 8.18 所 示 。 
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8.18 ”365 影视 音乐 网 后 台 首页 


8.9.2 后 台 首 页 技术 分 析 


在 后 台 的 主 显示 区 中 显示 了 各 个 模块 的 操作 及 结果 。 根 据 浏 览 器 地 址 栏 中 传递 的 不 同 的 参数 值 ， 
在 右 侧 显示 不 同 的 功能 界面 。 为 了 实现 这 个 功能 ， 在 程序 中 使 用 了 switch 语句 。switch 语句 和 让 语句 
类 似 ， 它 将 同一 个 表达 式 与 很 多 不 同 值 进行 比较 ， 当 获取 相同 的 值 时 执行 相应 的 代码 ，switch 语句 的 
语法 格式 如 下 : 
switch( 变 量 或 表达 式 { 
case 常量 表达 式 1: 
语句 1; 
break; 
case 常量 表达 式 2: 





case 常量 表达 式 n: 
语句 rm 
break; 
default: 
语句 n+1; 


switch 语句 根据 变量 或 表达 式 的 值 ， 依 次 与 case 中 的 常量 表达 式 的 值 相 比较 ， 如 果 不 相等 ， 继 续 
查找 下 一 个 case; 如 果 相 等 ， 就 执行 对 应 的 语句 ， 直 到 switch 语句 结束 或 遇 到 break 为 止 。 一 般 来 说 ， 


® 
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switch 语句 最 终 都 有 一 个 默认 值 default, 如 果 在 前 面 的 case 中 没有 找到 相符 的 条 件 ， 则 输出 默认 语句 。 
在 后 台 首页 中 ， 将 地 址 栏 传递 的 参数 action 的 值 作为 switch 语句 的 条 件 变量 ， 根 据 该 参数 的 不 同 
值 ， 应 用 include 语句 包含 相应 的 PHP 文件 ， 实 现 显示 不 同 操作 界面 的 功能 。 


8.9.3 ”后 台 首页 的 实现 过 程 


国 后台 首 页 使 用 的 数据 表 : tb manager 

本 系统 的 后 台 首 页 采用 的 布局 结构 为 二 分 栏 布局 。 左 侧 导 航 栏 清楚 地 显示 了 后 台 管 理 员 的 功能 。 
当 单 击 任意 功能 超 链接 时 ， 在 主 显示 区 显示 对 应 的 操作 界面 和 该 功能 模块 下 的 子 功能 。 页 面 简 练 、 结 
构 清晰 、 浏 览 方便 ， 二 分 栏 的 主要 特点 被 表现 得 淋漓 尽 致 。 

在 后 台 首 页 main.php 的 代码 中 ， 对 每 项 功能 权限 的 判断 都 是 在 leftphp 中 完成 的 ， 只 要 在 左 侧 导 
航 栏 的 位 置 载 入 leftphp 即 可 。 在 主 显 示 区 ， 应 用 switch 语句 根据 参数 返回 的 不 同 值 ， 显 示 不 同 的 功能 
界面 。 程 序 的 关键 代码 如 下 : 


倒 程 08 ”代码 位 置 ， 资源 包 \TM\08\online\admin\main.php 





<?php 

session_start(); /启动 Session 

include "inc/chec.php"; // 包 含 判断 用 户 权 限 文件 
include "conn/conn.php"; // 包 含 数据 库 连 接 文 件 

?> 

eee <!- 省 略 部 分 HTML 代码 --> 
<?php 

include "left.php"; /包含 文件 

?> 

< 上 -省 略 部 分 HTML 代码 --> 
<?php 


让 根据 不 同 的 参数 ， 显 示 不 同 的 功能 界面 */ 


switch (isset($_GET[action])?$_GET[action']:"){ 1/ 判断 参数 action 的 值 
case "audioList": // 如 果 参 数值 为 audioList 
include "a_list.php"; // 包 含 a_list.php 文 件 

break; /跳出 switch 语句 

case "videoList": // 如 果 参 数值 为 videoList 
include "v_list.php"; // 包 含 v_list.php 文 件 
break; /跳出 switch 语句 

case "audio": // 如 果 参 数值 为 audio 
include "audio.php"; /包含 audio.php 文件 
break:; /跳出 switch 语句 

case "video": /如果 参数 值 为 video 
include "video.php"; /包含 video.php 文件 
break; /跳出 switch 语句 

case "grade"™: // 如 果 参 数值 为 grade 
include "grade.php"; // 包 含 grade.php 文 件 
break; /跳出 switch 语句 

case "member // 如 果 参 数值 为 member 
include "member.php"; /包含 memberphp 文件 





局 
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break; /跳出 switch 语句 

case "log": // 如 果 参 数值 为 log 
include "log.php"; // 包 含 log.php 文 件 
break; // 跳 出 switch 语句 

case "manager": /| 如果 参数 值 为 manager 
include "manager.php"; /包含 manager.php 文件 
break: /跳出 switch 语句 

default: /默认 值 
include "a_list.php"; // 包 含 a_list.php 文件 
break; /跳出 switch 语句 


?> 


8.10 目录 管理 模块 设计 





8.10.1 目录 管理 模块 概述 


目录 管理 模块 主要 包括 视频 目录 管理 和 音频 目录 管理 两 个 部 分 。 管 理 员 可 以 通过 后 台 管 理 导 航 进 
入 对 应 的 目录 管理 页 面 。 

如 果 管 理 员 进入 了 视频 目录 管理 页 面 (videolistphp) ， 那 么 在 该 页 面 内 可 以 打开 目录 添加 页 面 进行 
目录 添加 操作 ， 也 可 以 删除 相应 的 目录 。 鉴 于 目录 信息 比较 简短 ， 本 系统 中 没有 提供 目录 修改 的 功能 。 


/A 
A a i le 
管理 模块 实现 的 方法 ， 关 于 音频 目录 管理 模块 的 实现 方法 请 参见 本 书 附 赠 资源 包 。 


8.10.2 ”目录 管理 模块 技术 分 析 


在 网 站 中 ， 有 个 功能 经 常 要 被 用 到 : 在 添加 目录 、 歌 曲 时 ， 经 常会 碰 到 重 名 的 问题 。 对 于 这 个 问 
题 ， 可 以 编写 一 个 自 定义 函数 ， 并 将 它 放 到 单独 的 一 个 文件 中 ， 方 便 其 他 页 面 调用 ， 如 本 系统 中 的 
“/inc/func.php”， 就 是 专门 用 来 存储 自 定义 函数 的 文件 。 下 面 来 看 一 下 用 来 判断 目录 是 否 重 名 的 自 定 
义 函 数 is_chk0， 代 码 如 下 : 


倒 程 09 代码 位 置 ， 资源 包 \TM\08\online\admin\inc\func.php 


// 判 断 目录 名 是 否 重复 
/1$f_fields: 字段 名 
//$tablename: 数据 表 名 
//$f_str: 要 查找 的 字段 
function is_chk($f fields,$tablename,$f_strj{ 
$is_chk = true; /初始 化 变量 
include "../conn/conn.php"; /包含 数据 库 连接 文件 
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Sis_sqlstr = "select $f _fields from S$tablename"; /定义 查询 语句 
Sresult = $pdo->prepare($is_sqlstr): /1 准备 查询 
Sresult->execute(); /执行 查询 
while($is_rst=$result->fetch(PDO::FETCH_NUM)X // 循 环 输出 查询 结果 
if($f_str == $is_rst[O]){ // 如 果 两 个 值 相等 
Sis_chk = false; /为 变量 赋值 
break; /跳出 循环 
} 
} 
return $is_chk; /| 返回 变量 的 值 





8.10.3 ”视频 目录 管理 的 实现 过 程 


国 ”目录 管理 使 用 的 数据 表 : tb_audiolist、tb_videolist 

单 击 左 侧 导航 栏 中 的 “视频 目录 管理 ” 超 链 接 ， 可 以 打开 视频 目录 管理 页 面 ， 在 管理 页 面 中 ， 有 
一 个 “目录 添加 ” 超 链接 ， 所 有 的 一 、 二 级 目录 信息 和 对 应 的 “删除 ” 超 链接 。 视 频 目录 管理 页 面 的 
运行 结果 如 图 8.19 所 示 。 





冰 议和 和 管理 
EEEEEREO 


等 和 


中 BY 
全 
可 
飞 





8.19 视频 目录 管理 页 面 


视频 目录 管理 页 面 的 代码 很 简单 : 首先 应 用 select 查询 语句 对 tb_videolist 数据 表 进 行 查询 ， 然 后 
使 用 while 循环 语句 循环 输出 查询 结果 ， 并 在 每 输出 一 条 记录 后 ， 添 加 一 个 “删除 ” 超 链 接 。 目 录 管理 
页 面 的 实现 代码 如 下 : 


倒 程 10 代码 位 置 资源 包 \TM\08\online\admin\v_listphp 








<?php 


include "inc/chec.php"; /包含 判断 用 户 权限 文件 





全 


第 8 章 365 影视 音乐 网 (Apache+PHP +phpMyAdmin+MySQL 5.5 实现 ) 





include "conn/conn.php"; /包含 数 据 库 连接 文件 
Sl_sqlstr = "select * from tb_videolist"; /定义 查询 语句 
$result = $pdo->prepare($L_sqlstn) /准备 查询 
Sresult->execute(); /执行 查询 

?> 


<table width="380" height="440" border="0" align="center" cellpadding="0" cellspacing="0"> 


<tr> 


<td colspan="4" valign="top"> 


</tr> 
</table> 


<table width="380" height="60" border="0" cellpadding="0" cellspacing="0"> 

<tr> 

<td height="20" colspan="4" align="center" valign="middle"> 视 频 目录 管理 </td> 
</tr> 
<tr> 

<td colspan="4"> 
<table width="375" border="0" align="center" cellpadding="2" cellspacing="2"> 
<tr> 





<td height="10" colspan="5" align="right" valign="middle"> 
<a href="#" onclick="javacript:Wopen=open('operation.php?action=videolist’, 
' 添 加 目录 ','height=500,width=665,scrollbars=no');"> 目 录 添加 </a> 
</td> 
</tr> 
<tr> 
<td height="30" align="center" valign="middle">ID</td> 
<td height="30" align="center" valign="middle"> 等 级 </td> 
<td height="30" align="center" valign="middle"> 名 称 </td> 
<td height="30" align="center" valign="middle"> 父 级 名 称 </td> 
<td height="30" align="center" valign="middle"> 操 作 </td> 
</tr> 
<?php 
while($l_rst=$result->fetch(PDO::FETCH_NUM)){// 循 环 输出 查询 结果 
?> 
<tr> 
<td height="18" align="center" valign="middle"><?php echo $I_rst[0]; ?></td> 
<td height="18" align="center valign="middle"><?php echo $l_rst[1]; ?></td> 
<td height="18" align="center valign="middle"><?php echo $l_rst[2]; ?></td> 
<td height="18" align="center valigl iddle"><?php echo $l_rst[3] ?></td> 
<td height="18" align="center" valign="middle"> 
<a href="del_list_chk.php?action=videolist&id=<?php echo $l_rst[0]; ?>" 
onclick="return del_chk();"> 删 除 </a> 






</td> 

</tr> 
<?php 

} 

?> 

</table></td> 
<ltr> 
</table></td> 
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8.10.4 视频 目录 添加 的 实现 过 程 


国 ”视频 目录 添加 使 用 的 数据 表 : tb audiolist、tb videolist 

单 击 视频 目录 页 面 中 的 “目录 添加 ”按钮 ， 
将 打开 视频 目录 添加 页 面 ,运行 结果 如 图 8.20 
所 示 。 

视频 目录 添加 页 面 (videolist.php) 中 主要 
包含 一 个 添加 目录 表单 ， 当 管理 员 添 加 信息 提 
交 后 ， 系 统 转 到 处 理 页 中 (videolist_chk.php) 
进行 添加 操作 。 首 先 判断 输入 目录 名 称 是 否 与 
已 存在 的 名 称 重复 ， 如 果 重 复 ， 回 到 上 一 步 。 
如 果 不 重复 ， 则 向 数据 表 中 添加 新 记录 。 实 现 
视频 目录 添加 操作 的 关键 代码 如 下 : 


视频 目录 书 各 洪 吉 


目录 名 各: 工人 上 


目录 直列: [三 可 同宗 


EEE LA LD 


区 名 医 本 
图 8.20 视频 目录 添加 页 面 





倒 程 11 代码 位 置 : 资源 包 \TM\08\online\admin\videolist_chk.php 





<?php 

header("Content-type: text/html; charset=utf-8"); 

session_start(); 

include "inc/chec.php"; 

include "conn/conn.php"; 

include "inc/func.php"; 

// 判 断 输入 的 名 称 与 数据 表 中 的 值 是 否 重复 

if(is_chk("name","tb_videolist",$_POST[names']) == falseX{ 
echo "<script>alert(' 名 称 重 复 ');history.go(-1);</script>"; 
exit(); 

} 

S$father = isset($_POST[father])?$_POST[father]:""; 


// 设 置 文件 编码 格式 
/启动 Session 

// 包 含 判断 用 户 权限 文件 
// 包 含 数据 库 连 接 文 件 

// 包 含 函 数 文件 


// 弹 出 对 话 框 
// 退 出 程序 


// 将 父 级 名 称 赋值 给 变量 


$a_sqlstr = "insert into tb_videolist (grade,name ,father,userName,issueDate) 
values(".$_POST[grade'].",".$_POST['names'.",".$father.", 


™.$_SESSION[admin].",".date("Y-m-d H:i:s").")"; 
if(I$pdo->exec($a_sqlstr)) 
echo "<script>alert( 添 加 失败 ');history.go(-1);</script>"; 
else 
echo "<script>top.opener.location.reload();alert(' 添 加 成 功 '); 
window.close();</script>"; 
?> 


8.10.5 ”目录 删除 的 实现 过 程 


国 ”目录 删除 使 用 的 数据 表 : tb_audiolist、tb_videolist 


/定义 插入 语句 
/如 果 执行 插入 语句 结果 为 假 
// 弹 出 对 话 框 


/弹出 对 话 框 


在 执行 目录 的 删除 页 面 del_list_chk.php 中 ， 首 先 根 据 地 址 栏 中 传递 的 action 参数 值 判断 要 删除 的 
目录 是 音频 目录 还 是 视频 目录 ， 然 后 根据 传递 的 目录 id 号 将 数据 库 中 对 应 的 记录 找到 并 删除 。 执 行 目 


录 删 除 的 关键 代码 如 下 : 


@ 
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倒 程 12 ”代码 位 置 : 资源 包 \TM\08\online\admin\del list_ chkphp 





<?php 

header("Content-type: text/html; charset=utf-8"); 1/ 设置 文件 编码 格式 

session_start(); /启动 Session 

include "inc/chec.php"; // 包 含 判断 用 户 权限 文件 

include "conn/conn.php"; // 包 含 数据 库 连接 文件 

if($_GET[action'] == "audiolist") // 如 果 参 数 action 的 值 为 audiolist 
$t_name = "tb_audiolist"; /为 变量 赋值 

else if($_GET[action'] == "videolist") /否则 如 果 参 数 action 的 值 为 videolist 
$t_name = "tb_videolist"; /为 变量 赋值 

$sqlstr = "delete from ".$t_name." where id ="$_GET[id]; /定义 删除 语句 

if(!$pdo->exec($sqlstr)\{ // 如 果 执行 删除 语句 结果 为 假 
echo "<script>alert(' 删 除 错误 !".$sqlstr.");history.go(-1);</script>";， ”// 弹 出 对 话 框 

jelse 
/弹出 对 话 框 


echo "<script>alert(' 删 除 成 功 ');location=".$_SERVER[HTTP_REFERER].";</script>"; 


?> 





8.11 数据 管理 模块 设计 


8.11.1 数据 管理 模块 概述 








数据 管理 模块 主要 包括 视频 数据 管理 和 音频 数据 管理 两 个 部 分 。 管 理 员 可 以 通过 后 台 管 理 导航 进 
入 对 应 的 数据 管理 页 面 ， 如 视频 数据 管理 页 面 (videophp) 。 在 数据 管理 页 面 ， 用 户 可 以 打开 数据 添 
加 页 面 或 删除 对 应 的 数据 。 视 频数 据 管理 页 面 的 运行 结果 如 图 8.21 所 示 。 


人 
CH 古寺 关山 帮 靳 下 二 和 刘磊 本 直到 


mu 夹 别 ES 损 作 
现 的 方法 基本 类 似 ， 因 此 ， 本 章 重 点 讲解 视频 数 
据 管理 模块 实现 的 方法 ， 关 于 音频 数据 管理 模块 2 六 
的 实现 方法 请 参见 本 书 附 赠 资源 包 。 时 sa 
5 3 
2 0 


8.11.2 ”数据 管理 模块 技术 分 析 





机 上 须 劲 漂 筷 理 





在 本 模块 中 ， 主 要 运用 的 就 是 文件 上 传 技术 。 通 
过 预定 义 变量 $_ FILES 可 以 判断 上 传 图 片 的 类 型 和 大 图 8.21 视频 数据 管理 页 面 
小 ， 并 实现 文件 的 上 传 操 作 。 这 里 使 用 了 一 个 自 定义 
函数 来 判断 文件 名 后 缀 。 实 现 过 程 如 下 : 


倒 程 13 ”代码 位 置 ; 资源 包 \TM\08\online\admin\inc\func.php 


// 判 断 文 件 后 缀 
/1$f_type: 允许 文件 的 后 组 类 型 
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/$f_upfiles: 上 传 文件 名 
function f_postfix($f_type,$f_upfiles){ 


$is_pass = false; /定义 变量 的 初始 值 为 false 
© S$tmp_upfiles = explode(".",$f_upfiles); /将 上 传 文件 名 以 “.” 为 分 割 符 进行 分 割 
© stmp_num = count($tmp_upfiles); /获取 分 割 后 生成 的 数组 元 素 的 个 数 


for($num = 0; $num < count($f_type):$num++}{ 
// 如 果 上 传 文件 后 缀 名 是 允许 的 文件 后 缀 类 型 


© if(strtolower($tmp_upfiles[$tmp_num - 1]) == $f_type[$num]){ 
$is_pass = 9f_type[$num]; // 将 后 缀 名 赋值 给 变量 
} 
} 
return $is_pass; 1/ 返回 变量 的 值 
} 
< 代码 贴 二 


@ explode(".",$f_ upfiles): explodeO) 函 数 为 字符 串 分 割 函数 ， 它 有 两 个 参数 ， 分 别 为 分 割 符 和 要 分 割 的 字符 串 。 返 
回 结果 为 分 割 后 的 字符 串 数 组 。 例 如 ，$tmp=explode("c","placepublicslice) ， 那 么 $tmp 值 就 为 Stmp[0]="pla"、 
$tmp[1]="epubli"、$tmp[3]="sli"、$tmp[4]="e"。 

@ count($tmp_ upfiles): countO 函 数 的 作用 是 取得 数组 个 数 。 

®@ strtolower($tmp_upfiles[$tmp_num - 1]): strtolowerO 函 数 返 回 参 数字 符 事 的 小 写 格 式 ， 如 strtolower("AbC")， 输 出 
等 于 abc。 


-/ 
Dg 
功能 实现 上 ， 都 无 法 和 PHP 自 带 的 函数 相 比较 。 


8.11.3 ”视频 数据 添加 的 实现 过 程 


国 ”数据 添加 使 用 的 数据 表 : tb audio、tb_video 
管理 员 可 以 通过 单 击 视频 数据 管理 页 面 的 
“数据 添加 ”按钮 打开 视频 数据 添加 页 面 
(videoadd.php ) ， 在 视频 数据 添加 页 面 填写 了 
表单 之 后 ， 单 击 “ 添 加 ”按钮 ， 将 表单 提交 给 
dataadd_chk.php 页 面 , 该 页 面 会 根据 提交 的 表单 
数据 在 数据 库 中 添加 对 应 的 记录 ,数据 添加 页 面 
的 运行 结果 如 图 8.22 所 示 。 
视频 添加 处 理 页 (dataadd_chk.php) 中 的 代 
码 主要 分 为 3 个 部 分 。 第 一 部 分 对 图 片 文件 格式 
及 图 片 大 小 进行 判断 , 如 果 正 确 无 误 则 执行 上 传 
操作 ; 第 二 部 分 对 视 / 音 频 文件 格式 及 文件 大 小 
进行 判断 ,如果 正确 无 误 则 执行 上 传 操作 ; 第 三 ED cm 
部 分 执行 向 数据 库 中 添加 新 记录 的 操作 。 图 8.22 ”视频 数据 添加 页 面 


@ 
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(1) 在 执行 图 片上 传 操作 时 ， 首 先 判 断 图 片 大 小 是 否 在 允许 范围 之 内 ， 接 着 使 用 自 定义 函数 
f_postfix() 来 判断 图 片 后 级 名 是 否 符合 要 求 ， 如 果 符 合 要 求 ， 则 根据 time0 函 数 生成 图 片 名 ， 并 保存 到 变 
量 $picture_path 中 , 再 应 用 move_uploaded file0 函 数 执行 文件 上 传 操作 。 实现 图 片上 传 操作 的 代码 如 下 : 


倒 程 14 代码 位 置 : 资源 包 \TM\08\online\admin\dataadd_chk.php 


header("Content-type: text/html; charset=utf-8"); // 设 置 文件 编码 格式 
session_start(); /启动 Session 

include "inc/chec.php"; // 包 含 判断 用 户 权限 文 件 
include "conn/conn.php"; // 包 含 数据 库 连 接 文 件 
include "inc/func.php"; /包含 函数 文件 

$p_type = array("jpg","jpeg","bmp","gif","png"); /定义 图 片 文 件 格式 数组 
/定义 音频 视频 文件 格式 数组 

$f_type = array("avi", "rm","rmvb","wav","mp3","mpg","mp4","wmv"); 

$audio_path = "../upfiles\audio"; /| 定义 上 传 音频 文件 路 径 
S$video_path = "../upfiles\\video"; /| 定义 上 传 视频 文件 路 径 
S$picture_path ="™"; /变量 初始 化 

Sfile_path = ™"; // 变 量 初始 化 


/判断 上 传 图 片 类 型 和 文件 大 小 ， 上 传 图 片 */ 

©@ if($_FILES['picture']['size'] >0 and $_FILES[picture']['size’] <700000){ 
// 如 果 图 片 格式 正确 

四 if($postf=f postfix($p_type,$_FILES['picture['name')) != falseX{ 








33 Spicture_path = time().".".$postf: /使 用 time() 函 数 生成 文件 名 
if($_POST[action] == "a"}{ // 如 果 action 值 为 a 时 ， 说 明 上 传 为 音频 图 片 
if($_FILES['picture' [tmp_name']) 
1/ 执行 上 传 操作 
9 move_uploaded_file($_FILES['picture'[tmp_name'],$audio_path."\".$picture_path); 
else{ 
// 弹 出 提示 对 话 框 
echo "<script>alert(' 上 传 图 片 失败 ! ');history.go(-1);</script>"; 
exit(); /1 退出 程序 
} 
}else if($_POST[action] == "v"}{ // 如 果 action 值 为 v 时 ， 说 明 上 传 为 视频 图 片 
这 $_FILES[picture][tmp_name]) 
/执行 上 传 操作 
move_uploaded file($_FILES[picture][tmp_name'],$video_path."\".$picture_path); 
else{ 
// 弹 出 提示 对 话 框 
echo "<script>alert( 上 传 图 片 失败 ! ');history.go(-1);</script>"; 
exit(); // 退 出 程序 
} 
} 
}else{ 
// 弹 出 提示 对 话 框 
echo "<script>alert(' 上 传 图 片 格式 错误 ! 111');history.go(-1);</script>"; 
exit(); // 退 出 程序 


} 
jelse if$_FILES[picturej[size] >700000){ 
echo "<script>alert( 上 传 图 片 大 小 超出 





');history.go(-1);</script>";// 弹 出 提示 对 话 框 


_ 丽 
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exit(); // 退 出 程序 


S$picture = ™"; 
> 





4 代码 贴 二 

@ 使 用 预定 义 变量 的 $ FILES[mame']['size'] 属 性 判断 上 传 文件 的 大 小 。 如 果 文 件 大 于 php.ini 中 设置 的 上 传 文件 的 最 
大 值 ， 那 么 size 将 永远 返回 0， 而 不 是 文件 的 实际 大 小 。 

固 使 用 自 定 义 函 数 了 postfix0 判 断 上 传 文件 的 后 缓 。 

目 返 回 当 前 UNIX 的 时 间 蕉 。 

@ 使 用 move Uploaded flleO 上 传 函 数 要 注意 : 它 的 第 一 个 参数 是 上 传 到 服务 器 中 的 临时 文件 名 ， 而 不 是 文件 原始 
名 称 ， 是 用 户 自 定义 的 名 称 。 要 取得 临时 文件 名 使 用 $_ FILES[name'][tmp_name'].。 


(2) 对 上 传 文件 的 判断 和 上 传 图 片 的 流程 基本 相同 ， 也 是 先 判断 文件 大 小 是 否 符合 上 传 文件 的 范 
围 要 求 ,接着 判断 上 传 文件 后 级 名 是 否 符合 要 求 ,最 后 使 用 time0 函 数 生成 文件 名 ,保存 到 变量 $file_path 
中 ， 并 应 用 move_uploaded file0) 函 数 执行 文件 上 传 操作 。 实 现 文件 上 传 操 作 的 代码 如 下 : 


倒 程 15 ”代码 位 置 ; 资源 包 \TM\08\online\admin\dataadd_chk.php 





/判断 上 传 文件 类 型 与 大 小 ， 上 传 文件 */ 
if($_FILES[address']['size’] >0X{ 


if($_POST[action"] == "v"X{ // 如 果 是 视频 文件 
if($_FILES['address']['size'] <300000000}X{ 
// 如 果 文件 格式 正确 
if(($postf = f_postfix($f_type,$_FILES[address'"[name')) != falseX{ 
Sfile_path = time().".".$postf; // 使 用 time() 函 数 生 成 文件 名 
if$_FILES['address][tmp_name]) 
// 执 行 上 传 操作 
move_uploaded _file($_FILES[address][tmp_name'],$video_path."N".$file_path); 
else{ 
// 弹 出 提示 对 话 框 
echo "<script>alert(' 上 传 文件 错误 ! ');history.go(-1);</script>"; 
exit(); // 退 出 程序 
» 
}else{ 
// 弹 出 提示 对 话 框 
echo "<script>alert( 上 传 文件 格式 错误 ! ');history.back(-1);</script>"; 
exit(); // 退 出 程序 
} 
jelse{ 
// 弹 出 提示 对 话 框 
echo "<script>alert( 上 传 文件 大 小 错误 ! ');history.go(-1);</script>"; 
exit(); // 退 出 程序 
} 
jelse if($_POST[action] == "a"}{ // 如 果 是 音频 文件 
if($_FILES[address'"['size’] <10000000X{ 
// 如 果 文件 格式 正确 


if(($postf = f_postfix($f_type,$_FILES[address [name')) != falseX{ 





@ 
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Sfile_path = time().".".$postf: 
if($_FILES['address"[tmp_name') 


// 执 行 上 传 操作 


// 使 用 time() 函 数 生成 文件 名 


move_uploaded file($_FILES[address'[tmp_name',$audio_path."\".$file_path); 


else{ 


1/ 弹出 提示 对 话 框 


echo "<script>alert(' 上 传 文件 错误 ! ');history.go(-1);</script>"; 


exit()' 
jelsef 


// 弹 出 提示 对 话 框 


// 退 出 程序 


echo "<script>alert(' 上 传 文件 格式 错误 ! '");history back(-1);</script>"'; 


exit(); 
上 
Jelse{ 
// 弹 出 提示 对 话 框 


// 退 出 程序 


echo "<script>alert( 上 传 文件 大 小 错误 ! ');history.go(-1);</script>"; 


exit(); 


} 
}else{ 
// 弹 出 提示 对 话 框 


// 退 出 程序 


echo "<script>alert( 没 有 上 传 文件 或 文件 大 于 300M');history.go(-1);</script>"; 


exit(); 
} 


// 退 出 程序 





(3) 如 果 上 传 图 片 和 上 传 的 文件 都 没有 问题 ， 则 对 表单 中 输入 的 其 他 信息 进行 处 理 ， 生 成 insert 
插入 语句 添加 新 记录 。 如 果 数 据 添加 成 功 则 刷新 父 窗口 ， 否 则 弹出 “添加 失败 ”的 信息 提示 框 。 执 行 


添加 操作 的 代码 如 下 : 


倒 程 16 ”代码 位 置 : 资源 包 \TM\08\online\admin\dataadd_chk.php 





$names =$_POST[names']; 


$grade = isset($_POST[grade])?$_POST[grade]:"; 


$sizes = $_FILES[address][size]; 
$publisher = $_POST[publisher]; 
$actor = $_POST[actor]; 

$language = $_POSTIIanguage]; 
$style = $_POST[style]; 

$type =$_POST[type]; 

S$from =$_POST[from1]; 

$publishtime = $_POST[publishtime]; 


$news = isset($_POST[news'])?$_POST[news']:™; 


S$remark = $_POST[remark]; 


S$intro = isset($_POST[Tintro])?$_POSTTintro]:"" 


J 

i($_POST[action] == "v")f 
S$director = $_POST[director]; 
Smarker =$_POST[marker]; 


// 获 取 数 据 名 称 

// 获 取 等 级 

// 获 取 上 传 文件 大 小 
// 获 取 发 行商 名 称 

// 获 取 主 演 或 歌手 名 称 
// 获 取 语 言 

// 获 取 二 级 分 类 

// 获 取 一 级 分 类 

// 获 取 发 行 国家 

// 获 取 发 行 时 间 

// 定 义 是 否 为 新 品 

// 获 取 简要 介绍 

// 获 取 详 细 介 绍 


// 如 果 上 传 文件 是 视频 文件 
/获取 导演 名 称 
/获取 制 片 人 名 称 
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$a_sqlstr = "insert into tb_video 


(name,picture,sizes,grade,publisher,actor,director,marker,languages,type, style, 
froms,publishtime,bool,remark,property,address,username,issueDate,intro) 
values('$names','$picture_path','$sizes','$grade','$publisher','$actor', 
'$director','$marker','$language','$type','$style','$from','$publishtime’, 
'$news",'$remark'",' 管 理 员 ','$file_path',”".$_SESSION['admin].", 


"date("Y-m-d H:i:s").",'$intro')"; /定义 插入 语句 
}else if($_POST[action] == "a"}X{ // 否 则 如 果 上 传 文件 是 音频 文件 
$actortype = $_POST[actortype]; // 获 取 歌 手 类 型 
$ci = $_POST[ci]; // 获 取 词 作者 名 称 
4$qu =$_POST[qu]; // 获 取 曲 作者 名 称 
/定义 插入 语句 


$a_sqlstr = "insert into tb_audio 


(name,picture,actor,ci,qu,actortype,type,style,publisher,froms,sizes,languages, 


publishTime,remark,property,address,userName,issueDate) 


values('$names','$picture_path','$actor'",'$ci",'$qu",'$actortype",'$type’, 
'$style','$publisher','$from','$sizes','$language','$publishtime','$remark’, 
管理 员 ','$file_path',".$_SESSION[admin].",".date("Y-m-d H:i:s").")"; 


Jelse{ 
echo "<script>alert(" 错 误 ');window.close();</script>"; // 弹 出 对 话 框 
exit(); // 退 出 程序 

} 

if($pdo->exec($a_sqlstr)) // 如 果 执 行 插入 语句 结果 为 真 
/弹出 对 话 框 


echo "<script>top.opener.location.reload();alert(' 添 加 成 功 ');window.close();</script>"; 


else 


echo "<script>alert(' 添 加 失败 ');history.go(-1);</script>"， ”// 弹 出 对 话 框 





通过 以 上 方法 就 可 以 实现 向 数据 库 中 添加 数据 的 功能 。 


a 


件 进行 设置 ， 否 则 将 无 法 实现 大 容量 文件 的 上 传 操 作 。 


设置 方法 如 下 : 

(1) 打开 phpStudy 的 主 界面 ， 如 图 8.23 所 示 。 

(2) 单 击 图 8.23 中 的 “其 他 选项 菜单 ”按钮 ， 然 后 依次 
选择 “PHP 扩展 及 设置 ”/“ 参 数值 设置 ”选项 ， 在 弹出 的 
选项 列表 中 需要 设置 两 个 选项 ， 一 个 是 post_max_size 选项 ， 
另 一 个 是 upload_max_filesize 选项 ， 如 图 8.24 所 示 。 

(3) 分 别 单 击 这 两 个 选项 会 弹出 相应 的 输入 框 ， 由 于 本 
项 目 中 上 传 文件 的 容量 范围 小 于 300M， 因 此 分 别 在 两 个 输入 
框 中 输入 “300M”， 如 图 8.25 和 图 8.26 所 示 。 输 入 完成 后 单 
击 “ 确 定 ” 按 钮 ，phpStudy 会 自动 重启 Apache 服务 器 使 新 的 
设置 生效 。 


370 
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8.11.4 ”数据 删除 


国 ”数据 删除 使 用 的 数据 表 : tb_audio、tb_video 
在 执行 视频 数据 删除 页 面 del_video_chk.php 中 , 根据 传递 的 视频 文件 的 id 值 查询 tb_video 数据 表 


中 该 视频 文件 的 上 传 文件 名 ， 然 后 应 用 站 语句 进行 判断 ， 如 果 该 文件 存在 则 使 用 unlinkO 函 数 删除 服务 
器 端的 真实 数据 ， 并 应 用 delete 语句 将 数据 库 中 的 该 条 视频 数据 删除 ， 程 序 代码 如 下 : 


倒 程 17 代码 位 置 , 资源 包 \TM\08\online\admin\del_video_chk.php 





<?php 

header("Content-type: text/html; charset=utf-8"); /1 设置 文件 编码 格式 
session_start(); /启动 Session 

include "conn/conn.php"; // 包 含 数 据 库 连接 文件 
include "inc/chec.php"; // 包 含 判断 用 户 权 限 文件 
Sfile_path = "../upfiles/video/"; /定义 视频 文件 路 径 


$s_sqlstr = "select * from tb_video where id = "$_GET[id]; /定义 查询 语句 


$result = $pdo->prepare($s_sqlstr); 1/ 准备 查询 

Sresult->execute(); /执行 查询 

if($s_rst=$result->fetch(PDO::FETCH_NUM)YX{ /将 查询 结果 返回 到 数组 并 判断 结果 是 否 为 真 
if(file_exists($file_path.$s_rst[16])X /1 如果 文件 存在 


/删除 文件 及 相应 图 片 并 判断 结果 是 否 为 真 
if(unlink($file_path.$s_rst[16]) and unlink($file_path.$s_rst[2])X{ 
$d_sqlstr = "delete from tb_video where id = ".$_GETT[id1]; /定义 删除 语句 
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if($pdo->exec($d_sqlstr)}{ // 如 果 执行 删 除 语句 结果 为 真 
// 弹 出 对 话 框 
echo "<script>alert(' 删 除 成 功 ');location='main.php?action=video';</script>"; 
exit(); // 退 出 程序 
Jelse{ 
echo "<script>alert(' 删 除 失败 ');history.go(-1);</script>"; // 弹 出 对 话 框 
exit(); // 退 出 程序 
上 
}else{ 
$d_sqlstr = "delete from tb_video where id = ".$_GETTid]; /定义 删除 语句 
讯 $pdo->exec($d_sqlstn))f // 如 果 执 行 删除 语句 结果 为 真 
// 弹 出 对 话 框 
echo "<script>alert(' 此 文件 已 删除 ~');location='main.php?action=video';</script>"; 
exit(); // 退 出 程序 
Jelse{ 
echo "<script>alert(' 删 除 失败 ');history.go(-1);</script>"; // 弹 出 对 话 框 
exit(); // 退 出 程序 
其 
} 
}else 
echo "<script>alert(' 删 除 失败 ');history.go(-1);</script>"; // 弹 出 对 话 框 


?> 





8.12 管理 员 设 置 模块 设计 








8.12.1 ”管理 员 设 置 模块 概述 


管理 员 设置 模块 主要 包括 管理 员 信息 添加 、 管 理 员 信息 删除 及 管理 员 信息 冻结 3 个 部 分 。 管 理 员 
可 以 通过 后 台 管 理 功能 导航 进入 管理 员 设置 页 
面 (managerphp )。 在 管理 员 设置 页 面 中 ， 管 
理 员 可 以 进行 管理 员 信息 的 添加 操作 ， 或 进行 也 
ef 
~、 
管理 员 


会 员 信息 的 冻结 或 删除 等 操作 。 管 理 员 设置 用 


例 图 如 图 8.27 所 示 。 
8.27 管理 员 设 置 用 例 图 





8.12.2 ”管理 员 设 置 模块 技术 分 析 


管理 员 可 以 单 击 管理 员 设 置 页 面 的 “冻结 ” 
或 “解冻 ”按钮 , 对 管理 员 信 息 进行 冻结 或 解冻 的 操作 。 当 管理 员 登 录 时 , 首先 查看 数据 库 中 的 whether 
字段 ， 如果 为 1， 则 说 明 该 用户 处 于 激活 状态 ; 当 单 击 “ 冻 结 ” 按钮 时 将 显示 “操作 成 功 ”的 提示 信息 。 
对 管理 员 信息 的 冻结 或 解冻 操作 主要 是 根据 传递 的 管理 员 的 ia， 将 数据 库 中 的 whether 字段 更 新 。 
程序 代码 如 下 : 


&@ 
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倒 程 18 ”代码 位 置 : 资源 包 \TM\08\online\admin\m freeze_chkphp 





<?php 

header("Content-type:text/html;charset=utf-8"); // 设 置 文件 编码 格式 

session_start(); /启动 Session 

include "conn/conn.php"; // 包 含 数据 库 连 接 文 件 

include "inc/chec.php"; // 包 含 判断 用 户 权 限 文件 

根据 whether 值 ， 进 行 取 反 操作 */ 

if($_POST[whether] == "1") // 如 果 传递 过 来 的 参数 whether 的 值 为 1 
$wt = "0"; /为 变量 赋值 

else if($_POST[whether] == "0") /否则 如 果 传递 过 来 的 参数 whether 的 值 为 0 
$wt = "1"; /为 变量 赋值 

else{ 
echo "<script>alert(' 非 法 操作 !");history.go(-1);</script>"; // 弹 出 对 话 框 
exit(); // 退 出 程序 


/根据 管理 员 id 和 赋值 后 的 $wt， 生 成 update 语句 */ 
$o_sqlstr = "update tb_manager set whether = ".$wt." where id = ".$_POSTT[id]; /定义 更 新 语句 


/执行 update 语句 */ 
if($pdo->exec($o_sqlstr)X{ // 如 果 执 行 更 新 语句 的 结果 为 真 
echo "<script>alert( 操作 成功 ');location='main.php?action=manager;</script>"; // 弹 出 对 话 框 
?> 
该 方法 实现 很 简单 ， 但 却 十 分 实用 ， 在 实行 会 员 制 的 网 站 上 ， 在 可 以 发 帖 、 回 复 的 论坛 中 ， 都 可 
以 使 用 这 个 方法 来 管理 。 


8.12.3 ”管理 员 添 加 的 实现 过 程 


国 。 管理 员 添加 使 用 的 数据 表 : tb manager 

管理 员 可 以 通过 管理 员 设置 页 面 打 开 管 理 员 信息 添加 页 面 (addmanagerphp )。 在 管理 员 信息 添加 
页 面 中 ， 管 理 员 可 以 填写 表单 并 将 表单 提交 到 处 理 页 进行 数据 处 理 ， 以 完成 管理 员 信息 添加 的 操作 。 
管理 员 添 加 页 面 的 运行 结果 如 图 8.28 所 示 。 








管理 员 信息 添加 


管理 员 名 帘 : videccata 
管理 本 码 : 
天 码 而 认 : 
管理 权限 : | 视频 数据 管理 员 
真实 姓名 : | 张无忌 

添加 ][ 返 回 


图 8.28 管理 员 添 加 页 面 的 运行 结果 
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管理 员 添加 页 面 涉及 的 HTML 表单 元 素 如 表 8.8 所 示 。 





表 8.8 管理 员 添 加 页 面 涉及 的 HTML 表单 元 素 




















名 称 类 型 重要 属性 含义 
list form method="post" action="addmanager chk.php" 表单 
names text id="names" size="30" 管理 员 名 称 
password password | id="password" size="30" 密码 
password2 password | id="password2" size="30" 密码 确认 
<option value=" 视 频 目 录 管 理 员 " selected> 视 频 目录 管理 员 </option> 
<option value=" 音 频 目录 管理 员 "> 音 频 目 录 管理 员 </option> 
<option value=" 视 频数 据 管理 员 "> 视 频数 据 管理 员 </option> 
gde Met a ac-" 井 磊 站 所 管理 员 => 首 姑 数 据 管理 员 <ptioup 管理 权限 
<option value=" 会 员 数 据 管理 员 "> 会 员 数 据 管理 员 </option> 
<option value=" 会 员 等 级 管理 员 "> 会 员 等 级 管理 员 </option> 
realname text id="realname" size="30" 真实 姓名 
Submit button class="submit" onClick="checkO:" “添加 ”按钮 
Submit2 button Value=" 返 ” 回 " class="submit" onClick="javascript:top.window.close()" “返回 ”按钮 





管理 员 信息 添加 的 数据 处 理 主要 是 根据 提交 的 表单 数据 ， 在 数据 库 中 添加 对 应 的 记录 ， 首 先 判断 
输入 的 名 称 是 否 存在 ， 如 果 存 在 ， 则 显示 提示 信息 ， 并 返回 到 上 一 步 ， 如 果 不 存 在 这 种 错误 ， 则 生成 
insert 语句 ， 添 加 新 管理 员 。 程 序 代码 如 下 : 


倒 程 19 代码 位 置 ， 资源 包 \TM\08\online\admin\addmanager_chk.php 





<?php 
header("Content-type:text/html;charset=utf-8"); // 设 置 文件 编码 格式 
session_start(); /启动 Session 
include "conn/conn.php"; // 包 含 数据 库 连 接 文 件 
include "inc/chec.php"; // 包 含 判断 用 户 权限 文 件 


$a_sql="select * from tb_manager where name=".$_POST[names].”"'; 


Sresult = $pdo->prepare($a_sql); // 准 备查 询 


$result->execute(); 


/执行 查询 


上 如 果 管 理 员 名 称 存在 ， 弹 出 提示 框 ， 并 返回 到 上 一 步 */ 


if($a_rst=S$result->fetch(PDO::FETCH_NUM)) 


/定义 查询 语句 


// 将 查询 结果 返回 到 数组 并 判断 结果 是 否 为 真 


echo "<script>alert(' 该 名 称 的 管理 员 已 经 存在 ， 请 更 换 名 称 ');history.go(-1);</script>";// 弹 出 对 话 框 
else{ 
/* 生 成 insert 语句 */ 

$a_sqlstr="insert into tb_manager values(",".$_POST[names'].",".$_POST[password].",".$_POST 


[grade].",".$_POST[realname'].",".date("Y-m-d").","1")"; 


让 执行 insert 语句 */ 
if($pdo->exec($a_sqlstr)}X{ 


<script> 
让 刷新 父 窗口 ， 弹 出 提示 框 ， 最 后 删除 当前 窗口 */ 
top.opener.location.reload(); 


alert(" 管 理 员 添加 成 功 "); 
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top.window.close(); 
</script> 
<?php 
} 


else 
echo "<script>alert(' 添 加 失败 ".$a_sqlstr."):history.go(-1);</script>"; 


8.12.4 ”管理 员 删 除 的 实现 过 程 


国 管理 员 删 除 使 用 的 数据 表 : tb_manager 
管理 员 信息 删除 主要 是 根据 传递 的 管理 员 的 id， 将 数据 库 中 对 应 的 数据 删除 。 程 序 代 码 如 下 ; 


倒 程 20 ”代码 位 置 ， 资源 包 \TM\08\online\admin\del mfreeze_chk.php 





<?php 
header("Content-type:text/html;charset=utf-8"); // 设 置 文件 编码 格式 
session_start(); /启动 Session 
include "conn/conn.php"; // 包 含 数据 库 连接 文件 
include "inc/chec.php"; // 包 含 判断 用 户 权限 文件 
$d_sqlstr = "delete from tb_manager where id = "$_GET[id]; /定义 删除 语句 
if($pdo->exec($d_sqlstr)) // 如 果 执 行 删除 语句 结果 为 真 
echo "<script>alert(' 删 除 成 功 ');location='main.php?action=manager";</script>";// 弹 出 对 话 框 
else 
echo "<script>alert(' 删 除 失败 ');history.go(-1);</script>";，”// 弹 出 对 话 框 
?> 





8.13 ”其 他 主要 功能 展示 





除了 上 面 详细 讲解 的 功能 模块 之 外 , 365 影视 音乐 网 中 还 有 
一 些 其 他 的 功能 ， 由 于 篇 幅 限制 ， 本 节 将 主要 展示 365 影视 音 
乐 网 中 其 他 主要 功能 的 效果 ， 具 体 实现 代码 请 参考 随 书 资源 包 
中 的 源码 文件 。 


8.13.1 用 户 注册 功能 


用 户 在 网 站 中 进行 在 线 观看 或 在 线 试听 之 前 必须 注册 为 网 
站 的 会 员 ， 在 注册 过 程 中 需要 用 户 填 写 用 户 名 、 密 码 、 密 码 提 
示 问 题 以 及 问题 答案 等 信息 。 用 户 注册 页 面 的 运行 结果 如 图 8.29 
所 示 。 
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图 8.29 用户 注 册页 面 
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8.13.2 用 户 登 录 功 能 


用 户 在 注册 完成 之 后 , 需要 先 登录 才能 进行 在 线 观看 或 在 线 试听 
等 操作 。 用 户 成 功 登 录 后 ， 在 页 面 右 上 方 会 显示 登录 的 用 户 名 、 会 员 
等 级 以 及 用 户头 像 等 信息 ， 如 图 8.30 所 示 。 


8.13.3 下 载 功 能 


365 影视 音乐 网 中 还 提供 了 视频 文件 或 音频 文件 下 载 的 功能 。 下 载 影片 或 下 载 歌 曲 的 功能 只 有 高 级 
会 员 登 录 后 才能 实现 ， 而 普通 会 员 只 有 在 线 观 看 或 在 线 试听 的 权限 。 在 本 项 目 中 ， 只 有 成 功 上 传 文件 5 
次 以 上 的 普通 会 员 才 会 升级 为 高 级 会 员 。 单 击 “ 下 载 ”按钮 后 的 运行 结果 如 图 8.31 所 示 。 
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8.30 用 户 登 录 后 的 显示 结果 



































HR 亲 建 xf 末 全" © 
言 ea fF 让 Adminisuetor 
mn 加 
a 寺 站 
WL a ee 人 Ri 
ED 365EHADHG 
司 subversion YXH 关 ZX 
un Wn7 mTRAaE ed 
四 [| xx ZW 闪 
EE 
Ne 四 net Lp ba 
mas EE 
人 BRICD [MP4 Video 
Smaps [ms 


图 8.31 选择 下 载 路 径 


说明 在 tb_grade 数据 表 中 可 以 对 上 传 文件 次 数 进行 设置 。 如 果 普 通 会 员 上 传 文件 次 数 达到 该 
值 ， 则 会 升级 为 高 级 会 员 。 


8.14 开发 技巧 与 难点 分 析 


在 本 系统 中 ， 多 个 地 方 使 用 了 无 边框 窗口 和 刷新 父 级 窗口 的 技术 。 下 面 来 学 习 有 关 这 两 方面 的 知 
识 要 点 。 


8.14.1 无 边框 窗口 
平时 看 到 的 浏览 器 窗口 都 是 有 边框 的 ， 如 果 偶 尔 有 一 个 无 边框 的 窗口 弹出 ， 那 么 将 会 更 加 引 人 注 
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目 。 实 现 无 边框 窗口 使 用 的 是 JavaScript 脚本 中 的 open0 方 法 ， 该 方法 的 常用 格式 如 下 : 


window.open(url,name ,features,replace); 





open() 方 法 中 的 参数 说 明 如 表 8.9 所 示 。 
表 8.9 open() 方 法 中 的 参数 说 明 


在 弹出 窗口 中 要 打开 的 地 址 

要 打开 窗口 的 名 字 。 在 使 用 target 属性 时 会 用 到 ， 可 以 为 空 
列举 的 窗口 特征 。 如 果 不 写 任何 参数 ， 那 么 默认 的 参数 多 数 情况 下 是 关闭 的 
一 个 boolean 值 ， 指 出 是 否 使 用 url 蔡 换 当前 内 容 。 该 参数 可 以 省 略 不 写 





name 





features 





replace 





下 面 应 用 window 对 象 的 open0 方 法 打开 一 个 指定 大 小 的 窗口 。 代 码 如 下 : 





<script language="javascript"> 
wopen = open("list.php","query","height=500,width=665,scrollbars=no,toolbar=yes,location=yes"); 
</script> 





除了 上 面 实例 中 给 出 的 几 个 窗口 特征 ， 表 8.10 还 列 出 了 其 他 几 个 常用 的 特征 参数 及 说 明 。 
表 8.10 浏览 器 窗口 的 外 观 样式 





属 性 值 说 明 
left 新 窗口 的 左 坐 标 
top 新 窗口 的 上 坐标 ， 配 合 lef 可 以 定位 一 个 窗口 的 弹出 位 置 
resizable 是 否 可 以 通过 拖 电 来 调整 新 窗口 的 大 小 ， 默 认为 no 
status 在 新 窗口 中 是 否 显示 状态 栏 ， 默 认为 no 


人 注意 证 笠 生 是 用 等 本 观 信 、 过 村 分 天 的 ， 在 过 写 丰 等 号 前 后 不 和 加 空 税 ， 因 为 在 有 由 浏览 
器 中 会 显示 错误 。 


8.14.2 ”刷新 父 级 窗口 和 关闭 子 窗口 


在 打开 的 子 窗口 进行 一 系列 的 操作 ， 如 添加 、 删 除数 据 后 ， 就 要 对 当前 窗口 进行 关闭 ， 最 后 返回 
父 级 窗口 中 ， 但 父 级 窗口 中 的 数据 可 能 仍然 是 没有 添加 或 删除 之 前 的 记录 数 ， 这 时 就 可 以 调 
JavaScript 脚本 对 父 窗口 进行 刷新 。 这 种 对 父 级 窗口 的 刷新 只 有 对 弹出 窗口 十 有效， 使 用 时 要 注意 。 该 
脚本 的 使 用 格式 如 下 : 




















top.opener.location.reload(); 


opener 属性 是 子 窗口 的 最 高 层 window 对 象 才 有 的 属性 ， 配合 location 对 象 可 以 对 父 窗口 进行 刷新 
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或 重 载 。location 对 象 的 两 个 常用 方法 有 href 方法 和 reload 方法 。href 方法 可 以 刷新 父 窗口 内 容 ， 使 用 
格式 如 下 : 


top.opener.location.href="operation.php"; 





关闭 子 窗口 使 用 如 下 代码 即 可 : 


Window.close(); 


不 过 这 里 要 注意 的 是 ，close0 方 法 在 关闭 子 窗口 时 , 不 弹出 提示 就 直接 关闭 浏览 器 , 但 对 于 父 级 窗 
口 ， 则 弹出 提示 框 ， 经 确认 后 才 可 以 关闭 浏览 器 。 


8.15 PDO 技术 专题 


PDO 是 PHP Date Object (PHP 数据 对 象 ) 的 简称 ， 它 是 与 PHP 5.1 版 本 一 起 发 行 的 ， 目 前 ， 大 多 
数 数据 库 都 支持 PDO。 有 了 PDO， 您 不 必 再 使 用 mysql * 函 数 、oci_ * 函 数 或 者 mssql * 函 数 ， 也 不 必 
再 为 它们 封装 数据 库 操作 类 ， 只 需要 使 用 PDO 接口 中 的 方法 就 可 以 对 数据 库 进 行 操作 。 在 选择 不 同 的 
数据 库 时 ， 只 需 修改 PDO 的 DSN (数据 源 名 称 )。 

在 PHP 6 中 将 默认 使 用 PDO 连接 数据 库 ， 所 有 非 PDO 扩展 将 会 在 PHP 6 中 被 移 除 。 该 扩展 提供 
PHP 内 置 类 PDO 来 对 数据 库 进行 访问 , 不同 数据 库 使 用 相同 的 方法 名 ， 以 解决 数据 库 连 接 不 统一 的 问 
题 。 下 面 介 绍 如 何 应 用 PDO 来 连接 和 操作 数据 库 。 


8.15.1 通过 PDO 连接 数据 库 


要 通过 PDO 连接 数据 库 ， 首 先 需 要 了 解 什么 是 DSN， 再 通过 PDO 的 构造 函数 来 连接 数据 库 ， 下 
面 分 别 进行 介绍 。 


1. DSN 详解 


DSN 是 Data Source Name (数据 源 名 称 ) 的 首 字母 缩写 。DSN 提供 连接 数据 库 需 要 的 信息 。PDO 
的 DSN 包括 3 部 分 : PDO 驱动 名 称 〈 如 mysql、sqlite 或 者 pgsql) 、 冒 号 和 驱动 特定 的 语法 。 每 种 数 
据 库 都 有 其 特定 的 驱动 语法 。 

在 使 用 不 同 的 数据 库 时 ， 必 须 明确 数据 库 服务 器 是 完全 独立 于 PHP 的 ， 是 实体 。 虽 然 在 讲解 本 书 
的 内 容 时 ， 数 据 库 服 务 器 和 Web 服务 器 是 在 同一 台 计算 机 上 ， 但 是 实际 的 情况 可 能 不 是 如 此 。 数 据 库 
服务 器 可 能 与 Web 服务 器 不 是 在 同一 台 计算 机 上 ， 此 时 要 通过 PDO 连接 数据 库 时 ， 就 需要 修改 DSN 
中 的 主机 名 称 。 

由 于 数据 库 服务 器 只 在 特定 的 端口 上 监听 连接 请 求 ， 故 每 种 数据 库 服务 器 具有 一 个 默认 的 端口 号 

(MySQL 是 3306) ， 但 是 数据 库 管理 员 可 以 对 端口 号 进行 修改 ， 因 此 有 可 能 PHP 找 不 到 数据 库 的 端 
口 ， 此 时 就 可 以 在 DSN 中 包含 端口 号 。 
另外 ， 由 于 一 个 数据 库 服务 器 中 可 能 拥有 多 个 数据 库 ， 所 以 在 通过 DSN 连接 数据 库 时 ， 通 常 都 包 
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括 数据 库 名 称 ， 这 样 可 以 确保 连接 的 是 您 想 要 的 数据 库 ， 而 不 是 其 他 的 数据 库 。 
2. PDO 的 构造 函数 
在 PDO 中 ， 要 建立 与 数据 库 的 连接 需要 实例 化 PDO 的 构造 函数 ，PDO 构造 函数 的 语法 如 下 : 
‘construct(string $dsn[.string Susernamel.string Spasswordl.array Sdriver optionsll) 


回 dsn: 数据 源 名 ,包括 主机 名 、 端 口号 和 数据 库 名 称 。 
回 username: 连接 数据 库 的 用 户 名 。 

回 ”password: 连接 数据 库 的 密码 。 

回 “driver_options: 连接 数据 库 的 其 他 选项 。 

例如 ， 通 过 PDO 连接 db_online 数据 库 的 代码 如 下 : 








<?php 
$dbms='mysql'; /数据 库 类 型 
$dbName='db_online' /使 用 的 数据 库 名 称 
$user=root'; /使 用 的 数据 库 用 户 名 
$pwd='root'; /使 用 的 数据 库 密码 
$host='"localhost'; // 使 用 的 主机 名 称 
S$dsn="$dbms:host=$host;dbname=$dbName"; 
try{ /捕获 异常 
$pdo=new PDO($dsn,$user$pwd); /实例 化 对 象 
echo "PDO 连接 MySQL 成 功 "; 
} catch (Exception $e) { 
echo $e->getMessage()."<br>"; 


} 


?> 





8.15.2 执行 SQL 语句 


在 PDO 中 ， 可 以 使 用 下 面 3 种 方法 来 执行 SQL 语句 。 
1. exec() 方 法 
exec() 方 法 返回 执行 后 受 影响 的 行 数 ， 其 语法 如 下 : 





int PDO::exec(string statement) 


参数 statement 是 要 执行 的 SQL 语句 。 该 方法 返回 执行 SQL 语句 后 受 影响 的 行 数 ， 通 常用 于 
INSERT、DELETE 和 UPDATE 语句 中 。 
例如 ， 在 PDO 中 通过 exec0 方 法 对 tb_grade 数据 表 执 行 更 新 操作 的 代码 如 下 : 


<?php 
Sdbms='mysql'; /数据 库 类 型 
S$dbName='db_online' /使 用 的 数据 库 名 称 
$user='root'; // 使 用 的 数据 库 用 户 名 
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?> 


S$pwd='root'; 
$host="localhost'; 
$dsn="$dbms:host=$host;:dbname=$dbName"; 
S$pdo=new PDO($dsn, $user,$pwd); 
$sql="update tb_grade set price=5 where id=1"; 
Sresult=$pdo->exec($sql); 
if($resultY{ 

echo "更 新 成 功 "; 
jelsef 

echo "更 新 失败 "; 
} 


2. query() 方 法 
query0 方 法 用 于 返回 执行 查询 后 的 结果 集 ， 其 语法 如 下 : 


/使 用 的 数据 库 密码 
/使 用 的 主机 名 称 


/实例 化 对 象 
/定义 更 新 语句 
/执行 更 新 语句 





PDOStatement PDO::query(string statement) 





参数 statement 是 要 执行 的 SQL 语句 。 它 返回 的 是 一 个 PDOStatement 对 象 。 
例如 ， 在 PDO 中 通过 query0 方 法 对 tb_videolist 数据 表 执 行 查询 的 代码 如 下 : 





<?php 


?> 


$dbms='mysql'; 
S$dbName='db_online'; 
S$user='root'; 
S$pwd='root'; 
$host='localhost'; 
$dsn="$dbms:host=$host;dbname=$dbName"; 
$pdo=new PDO($dsn, $user, $pwd); 
$query="select * from tb_videolist where grade=2"; 
Sresult=$pdo->query($query); 
foreach($result as $items){ 

echo $items['name’]." "; 


} 


/数据 库 类 型 

/使 用 的 数据 库 名称 
/使 用 的 数据 库 用 户 名 
/使 用 的 数据 库 密码 
/使 用 的 主机 名 称 


/实例 化 对 象 
/定义 SQL 语句 
/执行 查询 语句 ， 并 返回 结果 集 





3. 预 处 理 语句 
预 处 理 语句 包括 prepare0 和 execute0 两 个 方法 。 首 先 ， 通 过 prepare() 方 法 做 查询 的 准备 工作 ， 然 


后 ， 通 过 execute( 方 法 执行 查询 。 并 且 还 可 以 通过 bindParam() 方 法 来 绑 定 参数 提供 给 execute() 方 法 。 
其 语法 如 下 : 


PDOStatement PDO::prepare(string statement [, array driver_options]) 
bool PDOStatement::execute([array input_parameters]) 


例如 , 在 PDO 中 通过 预 处 理 语句 prepare0 和 execute() 方 法 对 tb_videolist 数据 表 执 行 查询 的 代码 


如 下 : 
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<?php 
S$dbms='mysql'; /数据库 类 型 
S$dbName='db_online': /使 用 的 数据 库 名 称 
$user=root'; /使 用 的 数据 库 用 户 名 
Spwd='root'; // 使 用 的 数据 库 密码 
$host="localhost’; // 使 用 的 主机 名 称 
$dsn="$dbms:host=$hostidbname=$dbName'"' 
$pdo=new PDO($dsn, $user,$pwd); /实例 化 对 象 
$query="select * from tb_videolist where grade=2"; /定义 SQL 语句 
$result=$pdo->prepare($query); // 准 备查 询 语句 
Sresult->execute(); /执行 查询 语句 ， 并 返回 结果 集 


while($res=$result->fetch(PDO::-FETCH_ASSOC)YX{ ll/ 循环 输出 查询 结果 集 ， 设 置 结果 集 为 关联 索引 
echo $res['name']." "; 


. 


?> 





8.15.3 ”获取 结果 集 


在 PDO 中 获取 结果 集 有 3 种 方法 : fetchO0、fetchAll0 和 fetchColumn0。 
1. fetch() 方 法 
fetch0 方 法 获取 结果 集中 的 下 一 行 ， 其 语法 格式 如 下 : 





mixed PDOStatement::fetch([int fetch_style [, int cursor_orientation [, int cursor_offset]]]) 





加 ”fetch_style: 控制 结果 集 的 返回 方式 ， 其 可 选 方式 如 表 8.11 所 示 。 
表 8.11 fetch_style 控制 结果 集 的 可 选 值 





值 说 明 
PDO::FETCH ASSOC 关联 数组 形式 
PDO::FETCH_ NUM 数字 索引 数组 形式 
PDO::FETCH BOTH 两 者 数组 形式 都 有 ， 这 是 默认 的 
PDO::FETCH_OBJ 按照 对 象 的 形式 ， 类 似 于 以 前 的 mysqli_fetch_object0 





PDO::FETCH_BOUND | ”以 布尔 值 的 形式 返回 结果 ， 同 时 将 获取 的 列 值 赋 给 bindParam0 方 法 中 指定 的 变量 
PDO::FETCH _ LAZY 以 关联 数组 、 数 字 索 引 数 组 和 对 象 3 种 形式 返回 结果 


加 ”cursor_orientation: PDOStatement 对 象 的 一 个 滚动 游标 ， 可 用 于 获取 指定 的 一 行 。 

加 ”cursor_offset: 游标 的 偏 移 量 。 

例如 ， 在 PDO 中 应 用 prepare0 和 execute() 方 法 对 tb_videolist 数据 表 执 行 查询 操作 ， 通 过 fetch0 
方法 返回 结果 集中 下 一 行 数据 , 同时 设置 结果 集 以 关联 数组 形式 返回 , 再 通过 while 语句 完成 数据 的 循 
环 输出 ， 代 码 如 下 : 


<?php 
S$dbms='mysql'; /数据 库 类 型 
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?> 


$dbName='db_online”; 

$user='root'; 

S$pwd='root'; 

$host="localhost'; 
$dsn="$dbms:host=$host;dbname=$dbName"; 
S$pdo=new PDO($dsn, $user, $pwd); 
$query="select * from tb_videolist where grade=2"; 
S$result=$pdo->prepare($query); 
Sresult->execute(); 


// 使 用 的 数据 库 名 称 
// 使 用 的 数据 库 用 户 名 
// 使 用 的 数据 库 密码 

// 使 用 的 主机 名 称 


// 实 例 化 对 象 

/定义 SQL 语句 

/准备 查询 语句 

/执行 查询 语句 ， 并 返回 结果 集 


while($res=$result->fetch(PDO::FETCH_ASSOC)X// 循 环 输出 查询 结果 集 , 并 且 设 置 结 果 集 以 关联 数组 形式 返回 


echo $res['name']." "; 


} 


2. fetchAll() 方 法 


fetchAll0 方 法 获取 结果 集中 的 所 有 行 ， 其 语法 如 下 : 


array PDOStatement::fetchAll([int fetch_style [, int column_index]]) 





回 
回 


其 返回 值 是 一 个 包含 结果 集中 所 有 数据 的 二 维 数组 。 


fetch_style: 控制 结果 集中 数据 的 显示 方式 。 
column_index: 字段 的 索引 。 


例如 , 在 PDO 中 应 用 prepare0 和 execute() 方 法 对 tb_videolist 数据 表 执 行 查询 操作 , 通过 fetchAll0 
方法 返回 结果 集中 的 所 有 行 ， 再 通过 for 语句 完成 结果 集中 所 有 数据 的 循环 输出 ， 代 码 如 下 : 





<?php 


和 > 


$dbms='mysql'; 
$dbName='db_online'; 
S$user='root'; 
S$pwd='root'; 
$host="localhost'; 
$dsn="$dbms:host=$host;dbname=$dbName"; 
S$pdo=new PDO($dsn, $user,$pwd); 
$query="select * from tb_videolist where grade=2"; 
$result=$pdo->prepare($query); 
$result->execute(); 
Sres=$result->fetchAlIl(PDO::FETCH_ASSOC); 
for($i=0;$i<count($res);$i++){ 

echo S$res[$i['name']." "; 


} 


/数据 库 类 型 

/使 用 的 数据 库 名称 
/使 用 的 数据 库 用 户 名 
/使 用 的 数据 库 密码 
/使 用 的 主机 名 称 


/实例 化 对 象 

/定义 SQL 语句 

/准备 查询 语句 

/执行 查询 语句 ， 并 返回 结果 集 
/获取 结果 集中 的 所 有 数据 
/循环 读 取 二 维 数组 中 的 数据 





3. fetchColumn() 方 法 
fetchColumn() 方 法 获取 结果 集中 下 一 行 指定 列 的 值 ， 其 语法 如 下 : 


string PDOStatement::fetchColumn([int column_numben) 


&@ 
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可 选 参 数 column_number 设置 行 中 列 的 索引 值 ， 该 值 从 0 开始 。 如 果 省 略 该 参数 则 将 从 第 1 列 开 
始 取 值 。 

例如 ， 应 用 prepare0 和 execute() 方 法 对 tb_videolist 数据 表 执 行 查询 操作 ， 再 通过 fetchColumn() 方 
法 输出 结果 集中 下 一 行 第 3 列 的 值 ， 代 码 如 下 : 


<?php 


$dbms='mysql'; /| 数据 库 类 型 
$dbName='db_online'; // 使 用 的 数据 库 名 称 
$user='root'; // 使 用 的 数据 库 用 户 名 
$pwd=root'; /使 用 的 数据 库 密码 
$host='"localhost'; // 使 用 的 主机 名 称 
S$dsn="$dbms:host=$host;dbname=$dbName"; 

$pdo=new PDO($dsn, $user,$pwd); // 实 例 化 对 象 

$query="select * from tb_videolist where grade=2"; /定义 SQL 语句 
Sresult=$pdo->prepare($query); // 准 备查 询 语句 
$result->execute(); /执行 查询 语句 ， 并 返回 结果 集 


echo $result->fetchColumn(2)." "; 

echo $result->fetchColumn(2)." "; 

echo $result->fetchColumn(2); 
?> 





8.16 本 章 总 结 


365 影视 音乐 网 实现 了 一 个 在 线 播放 平台 。 该 网 站 具备 一 个 影视 音乐 网 站 所 需要 的 所 有 的 基本 功 
能 ， 包 括 影视 和 音乐 文件 的 上 传 、 下 载 、 播 放 及 介绍 。 还 介绍 了 JavaScript 脚本 语言 的 一 些 实用 技巧 。 
希望 通过 本 章 的 学 习 ， 读 者 不 仅 可 以 建立 一 个 自己 的 影视 网 ， 而 且 可 以 熟练 使 用 PDO 来 操作 数据 库 。 


全 和 < 二 
草 


明日 科技 企业 网 站 


( ThinkPHP 3.2.3 实现 ) 


企业 网 站 系统 是 一 个 信息 化 B/S 架构 下 的 软件 ， 既 可 以 为 企业 进 
行 宣传 ， 也 可 以 为 企业 带 来 经 济 效益 ， 同 时 还 可 以 实现 企业 各 项 目 业 
务 的 信息 化 管理 。 信 息 化 管理 是 现代 社会 中 小 型 企业 稳步 发 展 的 必要 
条 件 ， 它 可 以 提高 企业 的 知名 度 ， 最 大 限度 地 减少 因为 广告 费用 而 增 
加 的 额外 开销 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

M 了 解 如 何 进 行 系统 分 析 

M 了 解数 据 库 设 计 流程 

Mm” 熟悉 搭建 系统 架构 的 方法 

wi 掌握 ThinkPHP 技术 的 应 用 

Wm 掌握 网 页 布局 

Wm 掌 栓 幻 灯 片 轮 播 效果 
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9.1 开发 背景 





| 视频 讲解 


企业 门户 网 是 一 个 连接 企业 内 部 和 外 部 的 网 站 ， 能 够 为 企业 提供 单一 性 的 信息 资 
源 访问 入 口 。 无 论 是 企业 的 员工 、 客 户 、 供 应 商 还 是 合作 伙伴 ， 都 可 以 通过 企业 门户 网 获得 个 性 化 的 
服务 。 在 了 解 企业 基本 信息 和 各 方面 资源 情况 的 基础 上 ， 为 企业 的 各 相关 领域 提供 便利 的 交流 平台 
以 达成 无 颖 集合 企业 内 容 、 商 务 与 社区 的 服务 效果 。 本 网 站 采用 国内 比较 知名 的 ThinkPHP 框架 开发 ， 
用 户 在 学 习 本 网 站 的 同时 ， 还 能 更 好 地 了 解 ThinkPHP 框架 的 相关 知识 。 


9.2 需求 分 析 


随 着 网 络 信息 时 代 的 高 速 发 展 ， 企 业已 经 不 能 再 单单 去 依靠 传统 的 推广 模式 去 打开 市 场 ， 而 更 多 
的 公司 和 个 人 已 经 认识 到 了 企业 网 站 的 重要 性 。 对 于 企业 来 说 ， 网 站 是 一 个 企业 发 展 的 第 二 生命 力 ， 
网 站 的 设计 就 是 体现 一 个 公司 文化 和 价值 的 最 好 彰显 方式 ， 成 败 由 细节 决定 。 

企业 网 站 的 重 中 之 重 就 是 网 站 的 搭建 及 前 端 设 计 ， 设 计 就 是 网 站 的 灵魂 和 生命 力 ， 需 要 用 细节 去 
成 就 。 企 业 网 站 需要 包括 前 后 和 后 台 ， 后 台 用 于 企业 管理 人 员 上 传 更 新 数据 ， 而 前 台 负 责 数 据 的 展现 ， 
将 企业 信息 、 企 业 新 闻 等 内 容 展现 给 用 户 。 用 户 通过 企业 官网 可 以 对 企业 有 更 深 的 了 解 ， 从 而 实现 商 
务 合作 。 


93 系统 分 析 


9.3.1 系统 目标 


根据 客户 提供 的 需求 和 对 实际 情况 的 考察 一 一 一 一 一 一 
与 分 析 ， 该 电子 商务 应 该 具备 如 下 特点 : 让 网 

界面 设计 简洁 、 友 好 、 美 观 大 方 。 

操作 简单 、 快 捷 方便 。 

数据 存储 安全 、 可 靠 。 

信息 分 类 清晰 、 准 确 。 I 
提供 灵活 、 方 便 的 权限 设置 功能 ， 使 系 
统管 理 明 确 。 


9.3.2 ”系统 功能 结构 


明日 科技 企业 网 站 分 为 两 个 部 分 ， 分 别 为 前 a 
台 和 后 台 ， 其 具体 功能 结构 如 图 9.1 所 示 。 图 9.1 系统 功能 





加 回回 罗 回 


局 亚 
六 可 上 发 耸 
豆 
济 湛 江油 或 
网 在 豆 剖 
浴 : 
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9.3.3 功能 预览 


明日 科技 企业 网 站 系统 由 多 个 页 面 组 成 ， 前 台 首 页 运行 效果 如 图 9.2 所 示 ， 企 业 简介 运行 效果 如 图 9.3 
所 示 ， 后 台 登 录 运 行 效果 如 图 9.4 所 示 ， 后 台 主 页 运行 效果 如 图 9.5 所 示 。 
,由 吉林 省 明日 科技 





做 深 做 专 做 到 位 


Dedicated 。 Speclalzed 。 Perfsct 





企业 简介 
| ji Company profles 
| i | 





图 9.3 企业 简介 运行 效果 图 9.4 后 台 登 录 运 行 效果 


全 





得 明日 科技 企业 网 站 
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和 入 人 反 近 站 吾 全 2 支持 是 区 的 县 大 动 














| 二 | 
图 9.5 后 台 主页 运行 效果 
9.3.4 ”系统 流程 图 
企业 门户 网 的 业务 流程 如 图 9.6 所 示 。 企业 简介 | | 新 闻 核心 争 力 | | 习 系 我 们 | 
* a T 
9.3.5 ”开发 环境 浏览 返回 浏览 返回 四 泛 四 
和 章 | 党 
在 开发 明日 科技 企业 网 站 时 ， 该 项 目 使 用 的 软件 一 a uli 
开发 环境 如 下 。 登录 返回 
回 ”操作 系统 : Windows 7 及 以 上 /Linux。 上 .| 
加 ”集成 开发 环境 : phpStudy。 操作 后 台 管 理 | 一 操作 
回 ”PHP 版 本 : PHP7。 | 站 本 + 
加 ”MySQL 图 形 化 管理 软件 : Navicat for MySQL。 新 闻 管理 一 返回 返回 一 | 管理 员 管理 
回 开发 工具 : PhpStorm 9.0。 图 96 业务 流程 图 
回 ThinkPHP 版 本 : 3.2.3。 
浏览 器 : 谷歌 浏览 器 。 
9.3.6 文件 夹 组 织 结构 


在 进行 网 站 开发 前 ， 首 先 要 规划 网 站 的 架构 。 也 
就 是 说 ， 建 立 多 个 文件 夹 对 各 个 功能 模块 进行 划分 ， 
实现 统一 管理 , 这 样 做 易于 网 站 的 开发 、 管理 和 维护 。 





可 mdexphp 一 一 一 一 一 一 一 一 一 一 前台 入 口 文件 


本 项 目 中 , 使 用 默认 的 ThinkPHP 目录 结构 , 将 Home 日 READMEmd 一 一 一 一 一 一 一 一 一 一 一 使 用 说 明 
文件 夹 作 为 前 台 模块 ，Admin 文件 夹 作为 后 台 模 块 ， SS 
如 图 9.7 所 示 。 图 9.7 文件 夹 组 织 结构 
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视频 讲解 


9.4 数据 库 设 计 





9.4.1 数据 库 分 析 
本 系统 采用 MySQL 作为 数据 库 ， 数 据 库 名 称 为 mrsoft， 其 数据 表 名 称 及 作用 如 表 9.1 所 示 。 
表 9.1 数据 库 表 结 构 
表 名 含 义 作 用 
mr member 管理 员 表 用 于 存储 管理 员 用 户 信 息 
RE 系统 变量 表 用 于 存储 系统 变量 信息 
mr auth group 权限 组 表 用 于 存储 权限 组 信息 
mr_auth group access 用 户 分 组 对 应 表 用 于 存储 用 户 分 组 对 应 信息 
mr auth rule 权限 规则 表 用 于 存储 权限 规则 信息 
mr _ category 分 类 表 用 于 存储 分 类 信息 
mr article 文章 表 用 于 存储 文章 信息 
mr devlog 开发 日 志 表 用 于 存储 开发 日 志 信息 
mr flash 焦点 图 表 用 于 存储 焦点 图 信息 
mr links 链接 表 用 于 存储 链接 信息 
mr log 日 志 表 用 于 日 志 信 息 





9.4.2 ”数据 库 逻 辑 设计 


1. 创建 数据 表 


由 于 篇 幅 所 限 ， 这 里 只 给 出 较 重 要 的 数据 表 的 部 分 字段 ， 完 整数 据 表 请 参见 本 书 附带 资源 包 。 


回 mr_ category (分 类 表 ) 
表 mr_category 用 于 保存 分 类 数据 信息 ， 其 结构 如 表 9.2 所 示 。 
































表 9.2 分 类 表 结 构 
字 段 名 数据 类 型 认 值 备注 
id int(11) 主键 
type tinyint(1) 0: 正常 ，1: 单 页 2: 外 链 
pid int(11) 父 茵 
name varchar(100) 分 类 名 称 
dir varchar(100) 目录 名 称 
seotitle varchar(200) SEO 标题 
keywords varchar(255) 关键 词 
description varchar(255) 描述 
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字 段 名 数据 类 型 上 徊 “法 
content | text | 内 容 

url | varchar(255) | 链接 地 址 
cattemplate | varchar(100) | 分 类 模板 
contemplate | varchar(100) | 内 容 模板 











o int(11) 








mr_article( 文 章 表 ) 
表 mr _article 用 于 存储 文章 信息 ， 其 结构 如 表 9.3 所 示 。 


表 9.3 文章 表 结构 
字段 名 | 数据 类 型 | 默认 值 | 允许 为 空 | 自动 递增 备注 
aid in | | wo | 
sid in | | wo | 
title varchar255) | | wo | 
seotitle varchar(255) | as | SEO 标题 
keywords varchar(255) | | 
description | varchar(255) | | 
thumbnail varchar(255) | | 

| | wo 


content text 
t mmiged| | xo | 时 间 
n muiged| 0o | xwo | 单 击 


加 ”mr_setting (系统 变量 表 ) 
表 mr_setting 用 于 存储 系统 变量 信息 ， 其 结构 如 表 9.4 所 示 。 








表 9.4 系统 变量 表 结 构 


数据 类 型 





varchar(100) 





varchar(255) 











type tinyint(1) 0: 系统 ，1: 自 定义 








varchar(255) 说 明 


2. 数据 库 连 接 相关 配置 
在 ThinkPHP 全 局 配置 文件 中 配置 数据 库 信 息 ， 有 具体 配置 如 下 : 


倒 程 01 代码 位 置 资源 包 \TM\09\Mrsoft\Application\Common\Conf\db.php 


return array( 
'DB_TYPE' => 'mysql， // 数 据 库 类 型 
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"DB_HOST' => '127.0.0.1'， // 服 务 器 地 址 


'DB_NAME' => 'mrsoft, /| 数据 库 名 
'DB_USER' => root， /用 户 名 
'DB_PWD' =>'root, /密码 
'DB_PORT' => 3306, /端口 
'DB_PREFIX' => mr ， /数据 库 表 前 缀 


'DB_CHARSET'=> "utf8'， /数据 库 编码 默认 采用 utf8 


9.5 前 台 首 页 设计 


蕊 | 


9.5.1 前 台 首 页 概述 





当 用 户 访问 明日 科技 企业 网 站 时 ， 首 先进 入 的 便 是 前 台 首页 。 前 台 首页 是 对 整个 网 站 总 体内 容 的 
概述 。 在 明日 科技 企业 网 站 的 前 台 首页 中 ， 主 要 包含 以 下 内 容 。 
导航 栏 ， 主 要 包括 “首页 ””“ 企 业 简 介 ””“ 新 闻 ”“ 核 心 竞争 力 ” 和 “联系 我 们 ”5 个 链接 。 
幻灯 片 轮 播 : 将 企业 宣传 图 片 以 幻灯 片 形 式 轮 播 展示 。 
功能 栏 。 主 要 用 于 展示 企业 的 业务 领域 。 
版 权 信息 : 显示 网 站 的 版 权 信 息 。 


9.5.2 前台 首 页 技术 分 析 


在 前 台 首页 中 ， 导 航 栏 作为 前 台 每 个 页 面 的 通用 部 分 ， 可 以 作为 一 个 独立 文件 ， 供 其 他 页 面 引 用 。 
版 权 信息 虽然 也 是 通用 部 分 ， 但 是 在 首页 中 为 能 够 快速 进入 后 台 ， 设 置 了 “后 台 ” 链 接 ， 而 其 他 页 面 
没有 该 链接 ， 所 以 该 版 权 信息 文件 只 属于 前 台 首页 。 功 能 栏 用 于 显示 企业 的 业务 领域 ， 为 了 达到 美观 
的 效果 ， 使 用 图 片 方式 展示 。 幻 灯 片 轮 播 作为 首页 的 重点 内 容 ， 需 要 能 够 在 后 台 进 行 管理 ， 包 括 增 、 
删 、 改 、 查 等 操作 。 


9.5.3 ”导航 栏 的 实现 过 程 


由 于 头 部 和 导航 栏 都 是 通用 部 分 , 可 以 分 别 将 其 作为 一 个 独立 文件 , 在 其 他 前 台 页 面 使 用 <include> 
标签 引用 ， 头 部 具体 代码 如 下 : 


倒 程 02 ”代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Home\View\Public\header.html 
<IDOCTYPE html> 
<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"> 
<meta name="Viewport" 


@ 


加 加 加 加 
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content="width=device-width, initial-scale=1, maximum-scale=1"> 

<title>{$Think.CONFIG.sitename}</title> 
<meta name="keywords" content="{$Think.CONFIG.keywords}"/> 
<meta name="description" content="{$Think.CONFIG.description}"/> 
<link href="_PUBLIC__/css/main.css" rel="stylesheet" type="text/css"> 
<link href="_PUBLIC_/css/container.css" rel="stylesheet" type="text/css"> 
<link href="”_PUBLIC_ /css/reset.css" rel="stylesheet" type="text/css"> 
<link href="_PUBLIC__/css/screen.css" rel="stylesheet" type="text/css"> 
<script src="__PUBLIC__/js/jquery.min.js"></script> 
<script src="__PUBLIC__/js/jquery-ui.min.js"></script> 
<script src="__PUBLIC__/is/fwsliderjs"></script> 
<script src="_PUBLIC__/js/tab.js"></script> 

</head> 

<body> 








以 上 代码 中 引入 了 所 有 资源 文件 , 其 中 “ PUBLIC ”会 被 替换 成 当前 网 站 的 公共 目录 即 “/Public/”。 
“{$Think.CONFIG.sitename} ”是 mr_setting 表 中 字段 “sitename” 的 值 ， 即 “明日 科技 有 限 公司 ”。 
导航 栏 包括 5 个 链接 ， 单 击 后 跳 转 到 相应 的 网 页 ， 具 体 代码 如 下 : 


倒 程 03 代码 位 置 资源 包 \TM\09\Mrsoft\App\Home\View\Public\nav.html 





< 导航 --> 
<div class="header_bg"> 
<div class="wrap"> 
<div class="header"> 
<div class="logo"> 
<a href="{:-U('index'")}"> 
<img src="_PUBLIC__/images/logo.png" alt="> 
</a> 
</div> 
<div class="pull-icon"> 
<a id="pull"></a> 


</div> 
<div class="cssmenu"> 
<ul> 
<li> 
<a href="{:U('index")})"> 首 页 </a> 
</li> 
<li> 
<a href="{:-U('about”)}"> 企 业 简 介 </a> 
</li> 
<li> 
<a href="{:-U('Inews")}"> 新 闻 </a> 
</li> 
<li> 
<a href="{:-U('core')}"> 核 心 竞争 力 </a> 
<Ili> 


<li class="last"> 
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<a href="{:U('contact')}"> 联 系 我 们 </a> 
</li> 
</ul> 

</div> 

< 上 -清除 浮动 -> 

<div class="clear"></div> 

</div> 
</div> 
</div> 


上 述 代 码 中 ， 使 用 TU 方法 实现 页 面 跳 转 。U 方法 是 ThinkPHP 内 置 方 法 ， 用 于 URL 的 动态 生成 。 

TU 方 法 的 定义 形式 : U( 地 址 表达 式 '[ 参 数 ],[' 伪 静态 后 组 ],[ 显 示 域 名 ]), 常用 形式 U( 地 址 表达 式 )， 
如 果 不 定义 模块 的 话 ， 就 表示 当前 模块 名 称 ， 如 果 不 定 义 控制 器 的 话 ， 则 表示 当前 控制 器 。 
U('Public/login") 生 成 Admin 模块 下 Public 控制 器 下 的 login 方法 的 url 地址 。 U(verify”) 生 成 Admin 模块 
下 Public 控制 器 下 的 login 方法 的 url 地 址 ， 等 价 于 U(CAdmin/Public/verify)。 





<img src="{:U('Admin/Public/verify')}" id="imgcode" Onclick="this.src=this.src+'?+Math.random()"/> 





等 价 于 





<img src="/app/index.php/Admin/Public/verify.htmI"” id="imgcode" 
Onclick="this.src=this.src+'?+Math.random()"/> 





在 首页 模板 文件 中 ， 引 入 头 部 和 导航 栏 的 关键 代码 如 下 : 
倒 程 04 代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Home\View\Index\index.html 





<!-- 头 部 -> 

<include file="Public/header" /> 
<!- 导 航 --> 

<include file="Public/nav" /> 





页 面 运行 效果 如 图 9.8 所 示 。 


, 员 吉林 省 明日 科技 有 限 公司 





图 9.8 ”网 站 导航 运行 效果 
9.5.4 幻灯 片 轮 播 的 实现 过 程 
国 商品 分 类 模块 使 用 的 数据 表 : mr flash 


1. 获取 幻灯 片 数 据 


幻灯 片 数据 存储 于 mr_flash 表 中 ， 需 要 从 该 表 中 筛选 出 所 有 “tite” 字 段 值 为 “banner” 的 数据 ， 
并 且 根据 序号 升序 排列 即 可 。 关 键 代码 如 下 : 


后 
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倒 程 05 代码 位 置 资源 包 \TM\09\Mrsoft\App\Home\Controller\IndexController.class.php 
<?php 





namespace Home\Controller; 
class IndexController extends ComController 


// 首 页 
public function index() 
{ 
/获取 banner 
$banners = M(flash')->where(arrayl'title'=>'banner))->order('o asc')->select(); 
S$this->assign('banners',$banners); 。 // 页 面 赋值 
S$this->display(); // 泻 染 模板 


} 


上 述 代 码 中 ， 使 用 where0 方 法 第 选 出 “title” 字 段 值 为 “banner” 的 数据 ， 然 后 使 用 order0 方 法 根 
据 序号 字段 “o” 进 行 “ase” 升 序 排列 ， 返 回 结果 为 一 个 数组 ， 赋 值 给 $banners。 接 着 使 用 assign0 方 
法 进行 页 面 赋 值 ， 最 后 使 用 display() 方 法 泻 染 模板 。 


2. 展示 幻灯 片 效果 


获取 完 分 类 数据 后 ， 接 下 来 就 需要 泻 染 模板 显示 数据 了 。 由 于 幻灯 片 数据 是 一 个 二 维 数组 ， 所 以 
使 用 <foreach> 标 签 遍历 获取 数据 即 可 。 关 键 代 码 如 下 : 


倒 程 06 代码 位 置 ， 资源 包 \TM\09\Shop\Application\Home\View\Public\header.html 
<!-- 轮 播 -> 
<div id="fwslider" style="height: 554px;"> 
<div class="slider_container"> 
<foreach name="banners" item="banner" key="k"> 
<div class="slide" style="opacity: 1; z-index: 0; display: none;"> 
<img id="img{$k}" src="{$banner['pic]}"> 
</div> 
</foreach> 
</div> 
<div class="timers" style="width: 180px;"></div> 
<div class="slidePrev" style="|left: Opx; top: 252px;"> 
<span></span> 
</div> 
<div class="slideNext" style="right: Opx; top: 252px; opacity: 0.5;"> 
<span></span> 
</div> 
</div> 
<!-- 轮 播 --> 
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上 述 代 码 中 ， 使 用 了 fwslider 幻灯 片 插件 。 使 用 该 插件 前 需要 引入 相应 的 JavaScript 文件 。 在 
header.html 头 部 文件 中 ， 已 经 引入 了 如 下 文件 : 
<script src=”_PUBLIC_Jis/jiqueryminjs"></script> 
<script src="__PUBLIC__/js/jquery-ui.min.js"></script> 
<script src=”_PUBLIC_/is/fwsliderjs"></script> 


此 时 ， 直 接 人 遍历 $banners， 然 后 获取 对 应 的 图 片 路 径 即 可 ， 运 行 结果 如 图 9.9 所 示 。 


bs) 核心 这 争 力 联系 我 们 





[一 倘 深 做 专 做 到 位 


Dedicated 。 Specialized Perfect 如 加 


按时 间 设 置 自 这 
动 切换 指示 牌 


9.9 约 灯 片 运行 效果 图 









9.6 ”新闻 模块 设计 


9.6.1 新 闻 模 块 概述 
新 闻 模块 是 网 站 之 中 最 传统 的 交流 模块 ， 现 在 的 大 部 分 网 站 都 需要 使 用 新 闻 模块 进行 网 站 信息 交 


流 。 在 新 闻 模块 中 ， 管 理 人 员 能 够 通过 后 台 进 行 新 闻 的 发 布 和 修改 ， 用 户 能 够 在 前 台 页 面 中 进行 新 闻 
的 访问 和 查询 。 


9.6.2 ”新 闻 模 块 技术 分 析 


新 闻 模块 包括 2 个 部 分 : 新 闻 列 表 页 和 新 闻 详情 页 。 在 新 闻 列 表 页 ， 以 表格 的 形式 展示 所 有 新 闻 
的 标题 、 发 布 时 间 和 详情 按钮 。 新 闻 列 表 页 的 数据 来 源 于 mr _article 表 中 分 类 为 新 闻 的 数据 。 而 新 闻 详 
情 页 则 是 通过 单 击 “ 详 情 ” 按 钮 ， 在 <a> 标 签 的 超 链接 中 添加 新 闻 ZD 实现 的 。 进 入 新 闻 详 情 页 后 ， 根 
据 新 闻 ID， 获 取 新 闻 内 容 。 


& 
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9.6.3 ”新 闻 列 表 页 的 实现 过 程 


回 新闻 列 表 页 使 用 的 数据 表 : mr article 
当 用 户 单 击 导航 栏 中 的 “新 闻 ”， 即 可 进入 新 闻 列 表 页 。 在 新 闻 列 表 页 ， 以 表格 的 形式 展示 所 有 
新 闻 的 标题 、 发 布 时 间 和 详情 按钮 。 


1. 获取 新 闻 列 表 数 据 


在 Home 模块 的 Index 控制 器 中 添加 news0 方 法 , 该 方法 用 于 获取 所 有 分 类 为 “新 闻 ” 的 文章 数据 ， 
在 所 有 数据 中 ， 只 获取 新 闻 ID、 新 闻 标 题 和 创建 时 间 3 个 字段 ， 具 体 代码 如 下 : 


倒 程 07 代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Home\Controller\IndexController.class.php 





<?php 
namespace Home\Controller; 
class IndexController extends ComController 


‘ 

// 企 业 新 闻 

public function news(){ 
$news = M('article')->where(array('sid'=>37))->field('aid ,title,t')->select();”// 获 取 分 类 为 “新 闻 " 的 数据 
S$this->assign('news', $news); // 页 面 赋值 
S$this->display(); // 泻 染 模板 

} 

} 





上 述 代码 中 使 用 了 field0 方 法 , 该 方法 属于 模型 的 连贯 操作 方法 之 一 , 主要 目的 是 标识 要 返回 或 者 
操作 的 字段 ， 可 以 用 于 查询 和 写 入 操作 。 


2. 展示 新 闻 列 表 效果 
在 View/Index 模板 目录 下 ， 创 建 news.html 文件 ， 遍 历 新 闻 列 表 数 据 ， 关 键 代 码 如 下 : 


倒 程 08 代码 位 置 资源 包 \TM\09\Mrsoft\App\Home\View\Indexnews.html 


<!- 头 部 --> 
<include file="Public/header" /> 
<!- 导 航 --> 
<include file="Public/nav" /> 
<!I--banner--> 
<div class="second_banner"> 
<img src="_PUBLIC__/images/3.gif" alt="> 
</div> 
<!I--//banner—> 
< 上 -新 闻 --> 
<div class="container"> 
<div class="left"> 
<div class="menu_plan"> 
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<div class="menu_title"> 公 司 动态 <br><span>news of company</span></div> 
<ul id="tab"> 
<liclass="active"><a href="#'> 公 司 新 闻 </a></li> 
</ul> 
</div> 
</div> 
<div class="right"> 
<div class="location > 
<span> 当 前 位 置 : <a href="javascript:void(0)" id="a"></a> 
<a href="#"> 公 司 新 闻 </a> 
</span> 
<div class="brief" id="b"><a href="#'> 公 司 新 闻 </a></div> 
</div> 
<div style=" font-size:14px; margin-top:53px; line-height:36px;"> 
<div id="tab_con"> 
<div id="tab_con_2" class="dis-n" style="display: block;"> 
<table style="margin-top:70px"> 
<tbody> 
<tr class="tt_bg"> 
<td> 新 闻 标题 </td> 
<td> 发 布 时 间 </td> 
<td> 详 情 </td> 
</tr> 
<foreach name="news" item="V"> 
<tr> 
<td>{$v['title]}</td> 
<td>{$v[t]ldate="Y-m-d",###}</td> 
<td> 
<a style="color:#3F862E" target="_blank" href="{:U('detail',array 
(news_id'=>$v[aid]))j"> 详 情 </a> 


</td> 
</tr> 
</foreach> 
</tbody> 
</table> 
</div> 
</div> 
</div> 

</div> 
</div> 
<!--// 新 闻 --> 
<!-- 底 部 -> 
<include file="Public/footer" /> 
<I--// 底 部 --> 





上 述 代 码 中 ，S$v['] 是 时 间 戳 形式 数据 (如 1513067402) ， 通 过 使 用 模板 函数 来 获取 标准 时 间 ( 如 
2017-12-12) 。 详 情 按钮 使 用 <a> 标 签 实现 页 面 跳 转 ，<a> 标 签 的 href 属性 通过 U 方法 传递 news_id 参 
数 。 新 闻 列 表 页 运行 效果 如 图 9.10 所 示 。 


@ 
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,加 吉林 省 明日 科技 有 限 公司 


核 心 竞争 力 联系 我 们 








| @ 单 击 “新 闻 ” 按钮 
S 公司 新 闻 
Company news 


公司 动态 公司 新 闻 
| mm | ee | @ am | 显示 新 闻 列表 信息 








图 9.10 新 闻 列 表 页 运行 效果 


9.6.4 ”新闻 详 情 页 的 实现 过 程 


日 ”新 闻 详情 页 使 用 的 数据 表 : Imr article 
1， 获取 新 闻 详 情 页 数据 


在 Home 模块 的 Index 控制 器 中 添加 detail0 方 法 ， 当 从 新 闻 列 表 页 跳 转 至 新 闻 详 情 页 时 ， 接 收 到 
传递 的 参数 news id， 根据 news_id 从 mr_article 表 中 获取 该 ID 的 数据 ， 具 体 代码 如 下 : 


倒 程 09 ”代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Home\Controller\IndexController.class.php 


/新闻 详情 

public function detail(){ 
Snews_id = I('news_id',0); /接收 ID 
Snews = M('article')->where(array('aid'=>$news_id))->find(); // 获 取 mr_article 表 数 据 
Sthis->assign('news', $news); // 变 量 赋值 
Sthis->display(); // 泻 染 模板 


上 述 代码 中 ,使 用 I 方法 来 接收 传递 的 新 闻 ID，I 方 法 是 ThinkPHP 用 于 更 加 方便 和 安全 的 获取 系 
统 输入 变量 。I(news_id',0) 语 句 表示 如 果 接 收 的 参数 news_id 不 存在 ， 则 默认 为 0。find0 方 法 只 会 返回 
第 一 条 记录 ， 即 $news 是 一 维 数组 。 
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2. 展示 新 闻 详情 页 效果 
在 View/Index 模板 目录 下 ， 创 建 detail.html 文件 ， 获 取 新 闻 详 情 数据 ， 关 键 代码 如 下 : 


倒 程 10 ”代码 位 置 资源 包 \TM\09\Mrsoft\App\Home\View\Index\detail.html 


<!- 头 部 -> 
<include file="Public/header" /> 
<!-- 导 航 --> 
<include file="Public/nav" /> 
<!--banner--> 
<div class="second_banner"> 
<img src=”_PUBLIC_/images/4.gif' alt=""> 
</div> 
<I--//banner—> 
<!- 新 闻 --> 
<div class="container"> 
<div class="left"> 
<div class="menu_plan"> 
<div class="menu _title"> 公 司 动态 <br><span>news of company</span></div> 
<ul id="tab"> 
<liclass="active"><a href="#'> 公 司 新 闻 </a></li> 
</ul> 
</div> 
</div> 
<div class="right"> 
<div class="location "> 
<span> 当 前 位 置 : <a href="javascript:void(0)" id="a"></a> 
<a href="#"> 公 司 新 闻 </a></span> 
<div class="brief id="b"> 
<a href="#"> 公 司 新 闻 </a> 
</div> 
</div> 
<div style="font-size: 14px; margin-top: 53px; line-height: 36px;"> 
<div id="tab_con"> 
<div id="tab_con_2" class="dis-n" style="display: block;"> 
<div class="content_main"> 
<br><h2 style="font-size:28px;text-align:center">{$news['title]}</h2> 
{$news['content]} 
</div> 
</div> 
</div> 
</div> 


<include file="Public/footer" /> 
<!-// 底 部 -> 





运行 结果 如 图 9.11 所 示 。 


局 
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公司 新 闻 当前 位 和 : 公司 新 闻 


编程 学 习 者 的 福音 ! ! ! 明日 学 院 APP 现 已 全 面 更 新 


! 明日 学 这 apP 现 已 全 面 更 新 。 新 下 本 增加 社区 喇 块 并 对 首页 布局 进行 整体 从 化。 测量 学 习 资 源 、 内 容 动 杰 

















图 9.11 新 闻 详 情 页 效果 


9.7 前 台 其 他 模块 设计 





9.7.1 其 他 概述 


前 台 首 页 中 还 包括 “企业 简介 ”、“ 核 心 竞争 力 ” 和 “联系 我 们 ”3 个 模块 ， 由 于 这 3 个 模块 实现 
功能 相似 ， 所 以 统称 为 其 他 模块 一 起 讲述 。 


9.7.2 ”其 他 模块 技术 分 析 


其 他 模块 数据 均 来 源 于 mr_article 表 ， 分 别 根据 mr_article 表 中 的 “sid” 字 段 进行 区 分 。 例 如 在 
mr_category 表 中 ，ID 为 36 的 分 类 是 企业 简介 ， 所 以 在 查找 mr_article 表 中 where 条 件 为 sid=36 的 数 
据 即 是 企业 文章 的 内 容 。 


9.7.3 联系 我 们 的 实现 过 程 


“联系 我 们 ”页 面包 括 “ 公 司 名 称 ”、“ 联 系 人 ”和 “电话 ”等 常用 信息 ， 所 以 将 这 些 信息 在 后 台 
配置 后 作为 系统 变量 输出 。 普 通 的 模板 变量 需要 首先 赋值 后 才能 在 模板 中 输出 , 但 是 系统 变量 则 不 需要 ， 
可 以 直接 在 模板 中 输出 ， 系 统 变 量 的 输出 通常 以 {SThink} 开 头 ，“ 联 系 我 们 ”页 面 的 关键 代码 如 下 : 


倒 程 11 代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Home\View\Index\contact.html 
<!-- 头 部 --> 

<include file="Public/header" /> 

< 上 -导航 一 > 

<include file="Public/nav" /> 

<l--banner--> 

<div class="second_banner"> 
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<img src="”_PUBLIC_ /5.gif alt="> 
</div> 
<!I--//banner—> 
<!-- 联 系 我 们 --> 
<div class="container"> 

<div class="left"> 

<div class="menu_plan"> 
<div class="menu_title"> 


联系 我 们 
<br> 
<span>Associate program</span> 

</div> 

<ul id="tab"> 
<lionclick="changeValue(this)" class="active"> 

<a href="#"> 联 系 我 们 </a> 

</l> 

</ul> 

</div> 
</div> 


<div class="right"> 
<div class="location "> 
<span> 当 前 位 置 : 
<a href="#"> 联 系 我 们 </a> 
</span> 
<div class="brief id="b"> 
<a href="#"> 联 系 我 们 </a> 
</div> 
</div> 
<div style="font-size: 14px; margin-top: 53px; line-height: 36px;"> 
<div id="tab_con"> 
<div id="tab_con_4" class="dis-n" style="display: block;"> 
<table class="contact"> 
<tbody> 
<tr> 
<td width="18%" class="ct_bg"> 
公司 名 称 
</td> 
<td> 
{$Think.CONFIG.company_name } 
</td> 
</tr> 
<tr> 
<td class="ct_bg"> 
联系 人 
</td> 
<td> 
{$Think.CONFIG.contact_name } 
</td> 


</tr> 
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<tr> 
<td class="ct_bg"> 
电话 
</td> 
<td> 
{$Think.CONFIG.contact_tel } 
</td> 
</tr> 
<tr> 
<td class="ct_bg"> 
邮箱 
</td> 
<td> 
{$Think.CONFIG.email } 
</td> 
</tr> 
<tr> 
<td class="ct_bg"> 


{$Think.CONFIG.address } 
</td> 
</tr> 
<tr> 
<td class="ct_bg"> 
邮编 
</td> 
<td> 
{$Think.CONFIG.zip_code } 
</td> 
</tr> 
<tr> 
<td class="ct_bg"> 
公司 主页 
</td> 
<td> 
{$Think.CONFIG.company_website } 
</td> 
</tr> 
</tbody> 
</table> 
<div style="text-align: center > 
<img src="_PUBLIC__/images/map.jpg" alt="> 
</div> 
</div> 
</div> 
</div> 
</div> 
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</div> 

<!--// 联 系 我 们 --> 

<!-- 底 部 --> 

<include file="Public/footer" /> 
<!-// 底 部 -> 





如 果 后 台 添 加 完 自 定义 变量 后 ， 前 台 页 面 没有 立即 生效 ， 请 单 击 后 台 导 航 栏 的 清除 缓存 图 标 ， 如 
9.12 所 示 ， 然 后 再 次 刷新 前 台 页 面 。“ 联 系 我 们 ”运行 结果 如 图 9.13 所 示 。 
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图 9.13 联系 我 们 页 面 效果 
9.8 ”后 台 登 录 模 块 设计 


9.8.1 后 台 登 录 模块 概述 





后 台 登 录 功 能 主要 是 对 管理 员 输入 的 账号 、 密 码 及 验证 码 进行 验证 。 验 证 内 容 包括 账号 、 密 码 是 
否 为 空 、 验 证 码 是 否 正确 、 用 户 名 和 密码 是 否 匹 配 等 。 如 果 验 证 成 功 ， 则 成 功 登 录 后 台 ， 和 否则 提示 错 
误 信息 。 


9.8.2 ”后 台 登 录 模 块 技术 分 析 


虽然 后 台 登 录 的 主要 功能 是 检测 管理 员 输 入 的 用 户 名 和 密码 是 否 正确 ， 但 是 在 验证 之 前 需要 编写 
其 他 附属 的 功能 ， 如 判断 管理 员 是 否 登录 、 生 成 和 检测 验证 码 等 。 


@ 
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1. 判断 是 否 登 录 


管理 员 访 问 后 台 时 ， 系 统 首先 判断 管理 员 是 否 登录 ， 如 果 登 录 则 进入 后 台 主 页 ， 否 则 跳 转 至 登录 
页 ， 提 示 管 理 员 登录 。 由 于 在 后 台 的 每 一 个 页 面 都 需要 判断 管理 员 是 否 登 录 ， 所 以 将 检测 登录 的 功能 
写 入 父 类 ComController 的 check login0 方 法 。 当 子 类 继承 ComController 父 类 时 ， 调 用 check login0 
方法 ， 执 行 检测 是 否 登 录 。check_login0 方 法 的 具体 代码 如 下 : 


倒 程 12 代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Qwadmin\Controller\ComController.class.php 
public function check_login(){ 


session_start(); /开启 Session 
S$flag = false; 1/ 初 始 化 flag 
$salt = C("COOKIE_SALT"); 1/ 获取 配 置 项 COOKIE_SALT 
Sip = get_client_ip(); // 获 取 IP 地 址 
S$ua = $_SERVER[HTTP_USER_AGENT1]; // 获 取 客 户 端的 浏览 器 和 操作 系统 信息 
$auth = cookie('auth’); // 获 取 Cookie 
Suid = session('uid'); /| 获取 Session 
if (Guid) { // 潮 断 用 户 ID 是 否 存在 
$user = M('member')->where(array('uid' => $uid))->find(); /查找 用 户 信息 
if ($user) { 
// 判 断 加 密 后 密码 
if ($auth == password($uid.$userfuser].$ip.$ua.$salt){ 
S$flag = true; /ag 标示 设置 为 true 
Sthis->USER = $user; /用 户 信息 赋值 
} 
} 
return $flag; // 返 回 flag 标识 


} 





2. 生成 和 检测 验证 码 


ThinkPHP 自身 封装 了 生成 验证 码 、 检 验 验 证 码 的 Verify 类 ， 使 得 对 验证 码 的 操作 非常 方便 快捷 。 
具体 实现 时 ， 需 要 调用 Verify 类 中 的 entry0 方 法 生成 验证 码 ， 调 用 Verify 类 中 的 check0 方 法 检验 验证 
码 ， 具 体 代码 如 下 : 


倒 程 13 代码 位 置 资源 包 \TM\09\Mrsoft\App\Qwadmin\Controller\LoginController.class.php 
/| 生成 验证 码 

public function verify() 

{ 





$config = array( 
YontSize' => 14, // 验 证 码 字体 大 小 
"ength' => 4, // 验 证 码 位 数 
useNoise' => false, /| 关闭 验证 码 杂 点 
imageW' => 100, 
imageH' => 30, 
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S$verify = new \Think\Verify($config); 
$verify->entry(login'); 


} 

// 检 测验 证 码 

function check_verify($code, $id = ") 
S$verify = new \Think\Verify(); 
return $verify->check($code, $id); 

} 


接 下 来 ， 在 登录 页 面 模 板 文件 中 展示 生成 的 验证 码 图 片 ， 关 键 代 码 如 下 : 
倒 程 14 ”代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Qwadmin\View\Login\index.html 





<form action="{:U('login/login")}" method="post"> 


© 


<fieldset> 


<label class="block clearfix"> 
<span class="block input-icon input-icon-right"> 
<input type="text" class="form-control" name="user" 
placeholder=" 用 户 名 "/> 
<i class="ace-icon fa fa-user"></i> 
</span> 
</label> 


<label class="block clearfix"> 
<span class="block input-icon input-icon-right"> 
<input type="password" class="form-control" name="password" 
placeholder=" 密 码 "/> 
<i class="ace-icon fa fa-lock"></i> 
</span> 
</label> 


<div class="space"></div> 
<label class="block clearfix"> 
<span class="block input-icon "> 
<span class="inline"><input type="text" class="form-control" 
name="verify" placeholder=" 验 证 码 " 
id="code" required/></span> 
<img style="cursor:pointer;" src="{:U('login/verify)}" 
width="100" height="30" title=" 看 不 清楚 ? 单 击 刷新 " 
onclick="this.src = (CU(loginverify)}?+new Date().getTime()"> 
</span> 
</label> 


<div class="space"></div> 


<div class="clearfix"> 
<label class="inline"> 
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<input type="checkbox" class="ace" name="remember"/> 
<span class="lbl"> 记 住 我 </span> 
</label> 


<button type="submit" 
class="width-35 pull-right btn btn-sm btn-primary > 
<i class="ace-icon fa fa-key"></i> 
<span class="bigger-110"> 登 录 </span> 
</button> 
</div> 


<div class="space-4"></div> 
</fieldset> 
</form> 


生成 验证 码 运行 结果 如 图 9.14 所 示 。 
9.8.3 ”后台 登 录 的 实现 过 程 , 


准备 工作 完成 后 ， 在 后 台 登 录 页 面 中 填写 用 户 名 、 密 码 和 ° 
验证 码 ， 添 加 “登录 ”按钮 ， 提 交 表单 到 login0 方 法 。loginO Es eat| 
方法 需要 先 检测 用 户 输入 的 信息 ， 然 后 判断 用 户 输入 的 用 户 名 
和 密码 是 否 匹 配 ， 如 果 匹 配 ， 将 用 户 ID 写 入 Sessin， 将 用 户 信 
息 写 入 Cookie， 并 且 写 入 操作 日 志 ， 最 后 跳 转 到 后 台 主 页 ， 否 - 
则 提示 错误 信息 。 图 9.14 生成 验证 码 效果 


黑 明日 科技 企业 网 站 








记 作 我 NN 


倒 程 15 ”代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Qwadmin\Controller\LoginController.class.php 





// 后 台 登 录 
public function login() 
{ 
/* 验 证 用 户 输入 **/ 
S$verify = isset($_POST[verify]) ? trim($_POST[verify]) : "; 
if (!$this->check_verify($verify, "login')) { 
$this->error( 验 证 码 错误 ! ', U("login/index")); 
} 


$username = isset($_POST[user]) ? tim($_POST[user]) : "; 
$password = isset($_POST[password']) ? password(trim($_POST[password])) : "; 
S$remember = isset($_POST[remember]) ? $_POST[remember] : 0; 
if ($username == ") { 
S$this->error(' 用 户 名 不 能 为 空 ! ", U("login/index")); 
} elseif ($password == ") { 
$this->error( 密码 必须 ! ", U("login/index")); 


和 
/查找 用 户 名 和 密码 是 否 匹 配 */ 
$model = M("Member"); 
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$user = $model->field(uid,user)->where(array('user => $username, ‘password' => $password))->find(); 


if ($user) { // 登 录 成 功 
$salt = C("COOKIE_SALT"); 
Sip = get_client_ip(); 
$ua = $_SERVER[HTTP_USER_AGENT:; 
session_start(); 
session('uid', $user[uid"]); 
/加 密 cookie 信息 
$auth = password($userfuid].$user[user].$ip.$ua.$salt); 
if (Sremember) { 


cookie('auth', $auth, 3600 * 24 * 365); ll 记 住 我 
}else{ 
cookie('auth', $auth); 
; 
addlog(' 登 录 成 功 。'); // 写 入 日 志 
$url = U('index/index'); 
header("Location: $url"); // 跳 转 到 后 台 主 页 
exit(0); 
} else{ /登录 失败 
addlog(' 登 录 失 败 。', $username); // 写 入 日 志 


S$this->error(' 登 录 失败 ， 请 重 试 ! ', U("login/index")); /提示 错误 信息 





9.9 后 台 管 理 模 块 设计 





和 


9.9.1 后 台 登 录 模 块 概述 


后 台 管理 模块 相对 于 前 台 要 复杂 得 多 。 前 台 页 面 是 将 相关 数据 信息 展示 给 用 户 ， 但 是 数据 通常 都 
是 在 后 台 进行 统一 管理 。 从 数据 库 角 度 来 说 ， 前 台 模 块 相 当 于 单一 的 数据 读 取 ， 而 后 台 模块 则 包括 了 
所 有 的 增 、 删 、 改 、 查 操作 。 后 台 管理 模块 主要 包含 以 下 内 容 。 

回 ”系统 设置 : 主要 包括 “ 自 定义 变量 “网 站 设置 ”和 “后 台 菜单 设置 ”等 。 

回 用户 及 用 户 组 : 主要 包括 用 户 管理 和 用 户 组 管理 ， 通 过 设置 用 户 和 用 户 组 ， 能 够 实现 用 户 权 

限 的 控制 。 

加 ”网 站 内 容 : 主要 包括 文章 管理 和 文章 分 类 管理 。 

回 ”其 他 功能 : 主要 包括 友情 链接 和 焦点 图 。 

回 ”个 人 中 心 : 主要 包括 个 人 资料 管理 和 退出 系统 。 

由 于 本 项 目 主要 使 用 文章 管理 系统 ， 所 以 重点 讲解 文章 管理 和 文章 分 类 ， 即 网 站 内 容 模块 。 运 行 
效果 如 图 9.15 所 示 。 


© 
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图 9.15 网 站 内 容 模块 


9.9.2 ”网 站 内 容 模块 技术 分 析 


网 站 内 容 模块 包括 分 类 管理 和 文章 管理 。 添 加 文章 时 需要 选择 文章 所 属 分 类 ， 所 以 在 设计 表 结 构 
时 ， 需 要 关联 mr_aticle 表 和 mr_category 表 ,， 即 mr_article 表 中 包含 的 sid 字段 值 就 是 mr_category 表 的 


id 字段 值 。 
9.9.3 文章 管理 的 实现 过 程 


1. 文章 列表 
文章 列表 页 面包 括 如 下 3 部 分 内 容 。 


回 ”顶部 搜索 框 : 可 以 根据 分 类 名 称 、 文 章 标题 、 发 布 时 间 排 序 进行 搜索 。 


回 ”文章 列表 : 展示 文章 的 所 属 分 类 、 标 题 以 及 发 布 时 间 。 
回 ”底部 分 页 : 根据 分 页 链接 显示 数据 。 


文章 列表 页 需要 同时 实现 以 上 3 部 分 内 容 ， 即 在 获取 文章 数据 的 同时 结合 搜索 和 分 页 。 对 于 搜索 ， 
需要 根据 搜索 条 件 联合 查询 。 对 于 分 页 ， 可 以 使 用 Page 类 来 实现 。 具 体 代码 如 下 : 


倒 程 16 ”代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Qwadmin\Controllen\AtticleController.class.php 


public function index($sid = 0, $p = 1) 
{ 
$p = intval($p) > 0 ? $p: 1; 
Sarticle = M('article’); 
S$pagesize = 3; 
Soffset = $pagesize * ($p - 1); 
$prefix = C('DB_PREFIX'); 
$sid = isset($_GET[sid]) ? $_GET[sid] : "; 
S$keyword = isset($_GET[keyword]) ? htmlentities($_GET[keyword]) : "; 
S$order = isset($_GET[order]) ? $_GET[order] :'DESC 
$where ='1=17 


1/ 判断 当前 页 码 

// 实 例 化 article 类 
// 每 页 数量 

1/ 计算 记录 偏 移 量 
1/ 获取 表 前 组 
/获取 分 类 ID 
/获取 关键 字 
/获取 排序 
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/根据 分 类 筛选 

if ($sid) { 
$sids_array = category_get_sons($sid); /获取 所 有 的 子 级 id 
$sids = implode(',', $sids_array); // 将 数组 拆 分 为 字符 串 
S$where .= "and {$prefix}article.sid in ($sids) "; 


} 
// 根 据 关键 字 筛 选 
if ($keyword) { 
S$where .= "and {$prefix}article.title like '%{$keyword}%' "; 


} 

// 默 认 按 照 时 间 降序 

$orderby = "t desce"; 
if ($order == "asc") { 


S$orderby = "t asc"; 
下 
// 获 取 栏 目 分 类 
$category = M(category')->field('id,pid,name'")->order('o asc')->select(); 
$tree = new Tree($category); /实例 化 树 型 类 
$str = "<option value=\$id\$selected>\$spacer\$name</option>"; /| 生成 的 形式 
$category = $tree->get_tree(0, $str, $sid); // 得 到 树 型 结构 
Sthis->assign('category', $category); /导航 
$count = $article->where($where)->count(); /获取 数量 
/筛选 文章 内 容 


Slist = $article->field("{$prefix}article.*,{$prefix}category.name")->where($where)->order($orderby) 
->join("{$prefix}category ON {$prefix}category.id = {$prefix}article.sid") 
->limit($offset . ',' . $pagesize)->select(); 


$page = newWThink\Page($count, $pagesize); /实例 化 分 页 类 
$page = $page->show(); // 调 用 分 页 方法 
Sthis->assign('"list', $list); // 页 面 赋值 
Sthis->assign('page', $page); // 输 出 分 页 
Sthis->display(); // 泻 染 模 板 


} 
接 下 来 演 染 文章 列表 页 模板 ， 文 件 代码 如 下 : 
倒 程 17 代码 位 置 资源 包 \TM\09\Mrsoft\App\Qwadmin\View\Atticle\index.html 








<!--/section:settings.box--> 
<div class="row"> 
<div class="col-xs-12"> 
<!--PAGE CONTENT BEGINS--> 
<div class="cf"> 
<form class="form-inline" action="" method="get"> 
<a class="btn btn-info" href="{:U('add')}" value=""> 新 增 </a> 
<label class="inline"> 所 属 分 类 </label> 
<select name="sid" class="form-control"> 
<option value="0">-- 分 类 --</option> 
{$category} 
</select> 


<label class="inline"> 文 章 标题 </label> 


[CN 
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<input type="text" name="keyword" value="{:I('keyword')}" class="form-control"> 


<label class="inline">&nbsp;&nbsp; 文 章 排序 ，</label> 
<select name="order class="form-control"> 
<option value="desc" <if condition="l(order) eq desc">selected</if>> 
发 布 时 间 降 序 </option> 
<option value="asc" <if condition="l(order) eq asc">selected</if> > 
发 布 时 间 升 序 </option> 
</select> 
<button type="submit" class="btn btn-purple btn-sm"> 
<span class="ace-icon fa fa-search icon-on-right bigger-110"></span> 
搜索 
</button> 
</form> 
</div> 
<div class="space-4"></div> 
<form id="form" method="post" action="{:U('del)}"> 
<table class="table table-striped table-bordered"> 
<thead> 
<tr> 
<th class="center"><input class="check-all" type="checkbox" value="></th> 
<th> 所 属 分 类 </th> 
<th class="col-xs-7"> 文 章 标题 </th> 
<th> 发 布 时 间 </th> 
<th> 操 作 </th> 
</tr> 
</thead> 
<tbody> 
<volist name="list" id="val"> 
<tr> 
<td class="center"><input class="aids" type="checkbox" name="aids[]" 
value="{$val['aid]}"></td> 
<td><a href="{:U('index',array('sid'=>$val['sid"]))}" 
title="{$val[name'"]}">{$val['name']}</a> 
</td> 
<td>{$val[title]j</td> 
<td>{$val[t]ldate="Y-m-d H:i:s",##}</td> 
<td><a href="{:U('edit',array('aid'=>$val['aid"]))}"><i 
class="ace-icon fa fa-pencil bigger-100"></i> 修 改 </a>&nbsp&nbsp;<a 
href="javascript:;" val="{:U('del',array('aids'=>$val['aid"]))}" class="del"><i 
class="ace-icon fa fa-trash-o bigger-100 red"></i> 删 除 </a></td> 
<ltr> 
</volist> 
</tbody> 
</table> 
</form> 
<div class="cf’> 
<input id="submit" class="btn btn-info" type="button" value=" 删 除 "> 
</div> 
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{$page} 
<!L--PAGE CONTENT ENDS--> 
</div><!--/.col--> 
</div><!--/.row--> 


文章 列表 页 全 部 数据 如 图 9.16 所 示 ， 根 据 分 类 和 文件 标题 筛选 条 件 搜索 后 的 数据 如 图 9.17 所 示 。 








国 > 5 > 六 一时 有 3 坟 ， 加 
HI 六 宁 ES 
育种 力 。。 二 和 Le 履 6 际 
WR 志和 区 新 法 关 提亲 功 驻 式 上 络 11 1 ed 6 
] 册 忆 由 且 了 /aa 第 手术 守之 作 过 所 下 载 ， 癌 二 二 凡 大 太 去 直 2 6 这 








图 9.16 全 部 数据 
"= bl -ea as | := me «== | 
所 属 分 类 文章 标 闫 内 布 了 侣 所 
新 闻 技术 若 妖 区 新 增 ' 受 党 提 后" 功 髓 今日 正式 上 法 | 1 2017-12-12 1651:09 ZE 埃 日 B 隐 





图 9.17 筛选 数据 


2. 新 增 文章 

在 添加 文章 时 , 需要 选择 文章 分 类 ,所 以 需要 获取 mr_category 表 中 全 部 分 类 数据 ,并 且 以 “ 树 形 ” 
的 方式 进行 展示 ， 添 加 分 类 的 控制 器 代码 如 下 : 

倒 程 18 ”代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Qwadmin\Controllen\ArticleController.class.php 

public function add() 


{ 
S$category = M('category')->field('id,pid,name')->order('o asc')->select(); ”// 获 取 所 有 分 类 
Stree = new Tree($category); /实例 化 属性 类 
$str = "<option value=\$id \$selected>\$spacer\$name</option>"; /| 生成 的 形式 
$category = $tree->get_tree(0, $str, 0); // 获 取 树 形 结构 
Sthis->assign('category', $category); /| 导航 
Sthis->display(form'); // 泻 染 页 面 

上 


新 增 文章 时 ， 需 要 填写 文章 内 容 ， 为 更 好 地 展现 文章 内 容 ， 使 用 富 文本 编辑 器 KindEditor 实现 该 
功能 。 此 外 ， 由 于 添加 文章 和 修改 文章 的 表单 内 容 相同 ， 所 以 演 染 同一 个 模板 form，form 模板 的 关键 
代码 如 下 : 

倒 程 19 ”代码 位 置 资源 包 \TM\09\Mrsoft\App\Qwadmin\View\Atticle\form.html 


<form class="form-horizontal" id="form" method="post" action="{:-U('update")}"> 
<!—-PAGE CONTENT BEGINS--> 
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<input type="hidden" name="aid" value="{$article.aid}" id="aid"/> 
<div class="form-group"> 
<label class="col-sm-1 control-label no-padding-right" for="form-field-0"> 
文章 分 类 </label> 
<div class="col-sm-9"> 
<select id="sid" name="sid" class="col-xs-10 col-sm-5"> 
<option value="0">-- 分 类 --</option> 
{$category} 
</select> 
<span class="help-inline col-xs-12 col-sm-7"> 
<span class="middle"> 选 择 所 属 分 类 。</span> 
</span> 
</div> 
</div> 
<div class="space-4"></div> 
<div class="form-group"> 
<label class="col-sm-1 control-label no-padding-right" for="form-field-1"> 
文章 标题 </label> 
<div class="col-sm-9"> 
<input type="text" name="title" id="title" placeholder=" 文 章 标题 " 
class="col-xs-10 col-sm-5" value="{$article['title]}"> 
<span class="help-inline col-xs-12 col-sm-7"> 
<span class="middle"> 文 章 标题 不 能 为 空 。</span> 
</span> 
</div> 
</div> 
<div class="form-group"> 
<label class="col-sm-1 control-label no-padding-right" for="form-field-1"> 
SEO 标题 </label> 
<div class="col-sm-9"> 
<input ="text" name="seotitle" id="seotitle" placeholder="SEO 标题 " 
class="col-xs-10 col-sm-5" value="{$article['seotitle]}"> 
<span class="help-inline col-xs-12 col-sm-7"> 
<span class="middle"> 如 果 设 置 SEO 标题 ， 将 会 在 IE 标题 栏 显示 SEO 标题 。</span> 
</span> 
</div> 
</div> 
<div class="space-4"></div> 
<div class="form-group"> 
<label class="col-sm-1 control-label no-padding-right" for="form-field-2"> 
关键 词 </label> 
<div class="col-sm-9"> 
<input type="text" name="keywords" id="keywords" placeholder=" 关 键 词 " 
class="col-xs-10 col-sm-5" value="{$article[keywords]}"> 
<span class="help-inline col-xs-12 col-sm-7"> 
<span class="middle"> 文 章 关键 词 。</span> 
</span> 
</div> 
</div> 
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<div class="space-4"></div> 
<div class="form-group"> 
<label class="col-sm-1 control-label no-padding-right" for="form-field-3"> 
文章 摘要 </label> 
<div class="col-sm-9"> 
<textarea name="description" id="description" placeholder=" 文 章 摘要 " 
Class="col-xs-10 col-sm-5" 
rows="5">{$article['description"]}</textarea> 
<span class="help-inline col-xs-12 col-sm-7"> 
<span class="middle"> 文 章 摘要 、 描 述 。</span> 
</span> 
</div> 
</div> 
<div class="space-4"></div> 
<div class="form-group"> 
<label class="col-sm-1 control-label no-padding-right" for="form-field-4"> 
缩 略 图 </label> 
<div class="col-sm-9"> 
<div class="col-xs-10 col-sm-5"> 
{:Uplmage("thumbnail",100,100,$article['thumbnail])} 
</div> 
<span class="help-inline col-xs-12 col-sm-7"> 
<span class="middle"> 仅 支持 jpg、gif、png、bmp、jpeg， 且 小 于 1MB。</span> 
</span> 
</div> 
</div> 
<div class="space-4"></div> 
<div class="form-group"> 
<label class="col-sm-1 control-label no-padding-right" for="form-field-2"> 
文章 内 容 </label> 
<div class="col-sm-9"> 
<textarea name="content" id="content" 
style="width:100%;height:400px;visibility:hidden;">{$article['content"]}</textarea> 





</div> 
</div> 
<div class="space-4"></div> 
<div class="col-md-offset-2 col-md-9"> 
<button class="btn btn-info submit" type="button"> 
<i class="icon-ok bigger-110"></i> 
提交 
</button> 
&nbsp; &nbsp; &nbsp; 
<button class="btn" type="reset"> 
<i class="icon-undo bigger-110"></i> 
重 置 
</button> 
</div> 
<!- PAGE CONTENT ENDS -> 
</form> 
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i 
运行 结果 如 图 9.18 所 示 。 
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图 9.18 添加 文章 


填写 完 文章 内 容 后 ， 单 击 “ 提 交 ” 按 钮 ， 需 要 检测 提交 内 容 。 例 如 ， 是 否 选择 了 分 类 ， 是 否 添加 了 
文章 标题 等 。 为 实现 友好 的 交互 效果 ， 本 项 目 使 用 Bootboxjs 插件 实现 该 功能 。 运行 效果 如 图 9.19 所 示 。 








图 9.19 Bootbox 提示 信息 


3. 编辑 文章 

在 文章 列表 页 ， 为 标题 右 侧 的 “修改 ”按钮 设置 一 个 <a> 标 签 ， 在 <a> 的 href 属性 中 设置 包含 文章 
了 D 的 链接 地 址 ， 代 码 如 下 : 

<td><a href="{:-U('edit',array('aid'=>$val['aid"]))}"> 


<i class="ace-icon fa fa-pencil bigger-100"></i> 修 改 </a>&nbsp&nbsp;<a> 
</td> 


@ 
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运行 结果 如 图 9.20 所 示 。 














日 Ri 入 类 文本 发 布 到 间 mE 
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名 刘 生 中 下 状 )ava 竺 一 季 改 主 之 作 交 二 下 载 要 所 基于 二 大 居于 所 可耻 2017-12-12 1630:58 Pa 


图 9.20 修改 文章 页 面 效果 





单 击 文章 标题 右 侧 的 “修改 ”按钮 ， 开 始 编辑 文章 ， 编 辑 文章 的 代码 如 下 : 


倒 程 20 ”代码 位 置 : 资源 包 \TM\09\Mrsoft\App\Qwadmin\Controller\ArticleController.class.php 


public function edit($aid) 


{ 


} 


运行 结果 如 图 9.21 所 示 。 
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$aid = intval($aid); /接收 文章 ID 
$article = M('article')->where('aid=" . $aid)->find(); /根据 ID 查找 文章 数据 
if ($article) { 
$category = M('category')->field("id,pid,name')->order('o asc')->select(); ”// 获 取 所 有 分 类 
S$tree = new Tree($category); /实例 化 树 型 类 
$str = "<option value=\$id \$selected>\$spacer\$name</option>"; /生成 的 形式 
$category = $tree->get_tree(0, $str, $article['sid"]); // 得 到 树 型 结构 
S$this->assign('category', $category); /| 导航 
S$this->assign('article', $article); // 页 面 赋值 
}else{ 
S$this->error(' 参 数 错误 ! '); 
} 
Sthis->display(form'); // 泻 染 模 板 
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三 资源 (包括 图 片 素材) 已 经 开放 下 载 了 , 欢 池 fi 完 吉 括 殷 动手 文 哮 的 司 学 下 堵 使 用 ~ 一 
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2017 年 11 月 9 日 唤 8 点 ， 宰 号 申 老 蚌 在 斗 钙 直播 平台 直播 了 使 用 Jw* 开 安 从 到 小 必 契 尖 戏 《四 ) 杰 龙 的 死亡 与 其 活 ,现在 直播 时 的 源 


中， 本 主 直 括 为 en 直 揪 革 一季 的 收 训 之 作 。 后 翅 斤 计 划 正 在 规划 中 ， 大 宗 有 什么 寺 求 ， 可 以 在 此 及 归 帖 ， 处 由 格式 为 : 编程 








图 9.21 编辑 文章 
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4. 删除 文章 


删除 文章 有 两 种 方式 : 单 选 删 除 和 多 选 删除 。 单 击 文章 标题 右 侧 的 “删除 ”按钮 ， 可 以 单 选 删 除 
文章 ;选中 右 侧 复 选 框 ， 单 击 下 方 的 “删除 ”按钮 ， 可 以 删除 选中 的 所 有 文章 。 单 选 删除 和 多 选 删除 
如 图 9.22 所 示 。 
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图 9.22 单 选 删 除 和 多 选 删除 


对 于 单 选 删 除 ， 页 面 提交 的 是 一 个 文章 ID， 数 据 类 型 是 整数 。 而 对 于 多 选 删除 ， 页 面 提交 的 是 多 
个 文章 ID， 数 据 类 型 是 数组 。 所 以 ， 需 要 对 两 种 情况 单独 处 理 。 删 除 文章 的 代码 如 下 ; 


倒 程 21 代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Qwadmin\Controller\ArticleController.class.php 





public function del() 
‘ 
$aids = isset($_REQUEST[aids]) ? $_REQUEST[aids] : false; 。”// 接 收文 章 ID 
if (Saids) { 
if (is_array($aids)) { // 多 选 删除 
$aids = implode(,, $aids); 
$map['aid] = array('in', $aids); 
}else{ // 单 选 删除 
$map = 'aid=' . $aids; 


} 

if (M('article')->where($map)->delete()) { /删除 数据 
addlog(' 删 除 文章 ，AID: '. $aids); // 写 入 日 志 
$this->success(' 恭 喜 ， 文 章 删除 成 功 ! 小 

}else{ 
Sthis->error(' 参 数 错误 ! "); 


} 
}else{ 
$this->error(' 参 数 错误 ! "); 


} 
} 


上 述 代 码 中 ， 对 于 多 选 删除 使 用 了 $map['aid'] = array('in', $aids) 形 式 ， 其 中 $aids 是 字符 串 型 数据 ， 
如 “12.3”。 删 除 的 SQL 语句 等 价 于 “delete from 
mr_article where aid in (1.2.3)”， 即 删除 aid 为 1、2、 
3 的 数据 。 此 外 , 删除 数据 前 需要 提示 管理 员 是 否 确 ite 
认 删 除 ， 如 果 单 击 “ 确 定 ”按钮 ， 则 在 这 行 删除 操 se 
作 ， 运 行 效果 如 图 9.23 所 示 。 





系统 提示 





图 9.23 ”删除 提示 
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9.10 开发 技巧 与 难点 分 析 


9.10.1 什么 是 单一 入 口 


单一 入 口 通常 是 指 一 个 项 目 或 者 应 用 具有 一 个 统一 《但 并 不 一 定 是 唯一 ) 的 入 口 文件 ， 也 就 是 说 项 
目的 所 有 功能 操作 都 是 通过 这 个 入 口 文件 进行 的 ， 并 且 入 口 文件 往往 是 第 一 步 被 执行 的 。 单 一 入 口 的 好 
处 是 项 目 整体 比较 规范 ， 因 为 同一 个 入 口 ， 其 不 同 操作 之 间 往 往 具 有 相同 的 规则 。 另 外 一 个 方面 就 是 单 
一 入 口 控制 较为 灵活 ， 因 为 拦截 方便 ， 类 似 一 些 权限 控制 、 用 户 登录 方面 的 判断 和 操作 可 以 统一 处 理 。 


9.10.2 为 什么 要 使 用 MVC 设计 模式 


应 用 程序 中 用 来 完成 任务 的 代码 一 一 模型 层 (也 叫 “ 业 务 逻 辑 ”) ， 通 常 是 程序 中 相对 稳定 的 部 
分 ， 重 用 率 高 ， 而 与 用 户 交 互 界 面 一 一 视图 层 ， 却 经 常 改 变 。 如 果 因 需求 变动 而 不 得 不 对 业务 逻辑 代 
码 修改 ， 或 者 要 在 不 同 的 模块 中 应 用 到 相同 的 功能 而 重复 编写 业务 逻辑 代码 ， 不 仅 降低 整体 程序 开发 
的 进度 ， 也 会 使 未 来 的 维护 变 得 非常 困难 。 因 此 将 业务 逻辑 代码 与 外 观 分 离 ， 将 会 更 方便 地 根据 需求 
改进 程序 ， 所 以 通常 使 用 MVC 设计 模式 。 


9.10.3 ”清空 缓存 


由 于 存在 页 面 缓存 ， 当 在 后 台 配 置 完 系统 变量 后 ， 前 台 联系 我 们 页 面 可 能 不 会 马上 生效 ， 此 时 可 
以 单 击 后 台 的 清除 缓存 图 标 来 清除 缓存 。 清 除 缓存 的 关键 代码 如 下 : 


倒 程 22 ”代码 位 置 ， 资源 包 \TM\09\Mrsoft\App\Qwadmin\Controllen\AtticleController.class.php 








class CacheController extends ComController 
{ 


// 清 除 缓存 

public function clear() 

{ 
$cache =\Think\Cache::getlnstance(); // 实 例 化 缓存 类 
$cache->clear(); // 清 室 缓 存 
Sthis->rmdir(RUNTIME_PATH); /删除 缓存 文件 


$this->success(' 系 统 缓存 清除 成 功 ! '); 


9.11 ThinkPHP 视图 技术 专题 


在 ThinkPHP 里 面 ， 视 图 由 两 个 部 分 组 成 : View 类 和 模板 文件 。Action 控制 器 直接 与 View 视图 
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类 进行 交互 ， 把 要 输出 的 数据 通过 模板 变量 赋值 的 方式 传递 到 视图 类 ， 而 具体 的 输出 工作 则 交 由 View 
视图 类 来 进行 ， 同 时 视图 类 还 完成 了 一 些 辅助 的 工作 ， 包 括 调用 模板 引擎 、 布 局 泻 染 、 输 出 替换 、 页 
面 Trace 等 功能 。 为 了 方便 使 用 , 在 Action 类 中 封装 了 View 类 的 一 些 输出 方法 , 例如 display0、fetchO、 
assign0、trace0 和 buildHtml0 等 方法 ， 这 些 方法 的 原型 都 在 View 视图 类 里 面 。 


9.11.1 模板 定义 

每 个 模块 的 模板 文件 都 是 独立 的 ， 为 了 对 模板 文件 更 加 有 效 地 管理 ，ThinkPHP 对 模板 文件 进行 目 
录 划 分 ， 默 认 的 模板 文件 定义 规则 是 : 

视图 目录 /模板 主题 /控制 器 名 /操作 名 + 模板 后 级 


默认 的 视图 目录 是 模块 的 View 目录 (模块 可 以 有 多 个 视图 文件 目录 ， 这 取决 于 你 的 应 用 需要 ) ， 
框架 的 默认 视图 文件 后 级 是 .html。 新 版 模板 主题 默认 是 空 〈 表 示 不 启用 模板 主题 功能 ) 。 

在 每 个 模板 主题 下 ， 都 是 以 模块 下 面 的 控制 器 名 为 目录 ， 然 后 是 每 个 控制 器 的 具体 操作 模板 文件 ， 
例如 ，User 控制 器 的 add 操作 对 应 的 模板 文件 就 应 该 是 “./Application/Home/View/User/add.html“， 如 
果 默 认 视图 层 不 是 View， 例 如 : 


'DEFAULT_V_LAYER' => "Template', /设置 默认 的 视图 层 名 称 











那么 ， 对 应 的 模板 文件 就 变 成 了 “./Application/Home/Template/User/add.html”。 模 板 文 件 的 默认 
后 缀 是 .html， 也 可 以 通过 TMPL_TEMPLATE_SUFFIX 更 改 为 其 他 的 文件 名 。 例 如 : 





'TMPL_TEMPLATE_SUFFIX'=>".tpl' 


定义 后 ，User 控制 器 的 add 操作 对 应 的 模板 文件 就 变 成 “./Application/Home/View/User/add.tpl”。 如 
果 觉 得 目录 结构 太 深 ， 可 以 通过 设置 TMPL _FILE_DEPR 参数 来 配置 简化 模板 的 目录 层次 ， 例 如 设置 : 


TMPL_FILE_DEPR'=>'， 





默认 的 模板 文件 就 变 成 了 “./Application/Home/View/User_add.html”。 
9.11.2 ”模板 赋值 


如 果 要 在 模板 中 输出 变量 ， 必 须 在 控制 器 中 把 变量 传递 给 模板 ， 系 统 提 供 了 assign0 方 法 对 模板 变 
量 赋值 ， 无 论 何 种 变量 类 型 都 统一 使 用 assign 赋值 。 
S$this->assign('name', $value); 


/下 面 的 写法 是 等 效 的 


Sthis->name = $value; 
Assign() 方 法 必须 在 display0 和 show0 方 法 之 前 调用 ， 并 且 系 统 只 会 输出 设 定 的 变量 ， 其 他 变量 不 


会 输出 系统 变量 例外 ) ， 一 定 程度 上 保证 了 变量 的 安全 性 。 赋 值 后 ， 就 可 以 在 模板 文件 中 输出 变量 
了 ， 如 果 使 用 的 是 内 置 模板 的 话 ， 就 可 以 这 样 输出 : {$name}。 
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如 果 要 同时 输出 多 个 模板 变量 ， 可 以 使 用 下 面 的 方式 : 


$array[name] = ‘thinkphp’; 
$array[email] = "iu21st@gmail.com'; 
$array[phone] = '12335678"; 
S$this->assign($array); 


这 样 ， 就 可 以 在 模板 文件 中 同时 输出 name、email 和 phone 3 个 变量 。 模 板 变量 的 输出 根据 不 同 的 


模板 引擎 有 不 同 的 方法 ， 在 后 面 会 专门 讲解 内 置 模板 引擎 的 用 法 。 如 果 使 用 PHP 本身 作为 模板 引擎 的 


话 ， 


9:1 


就 可 以 直接 在 模板 文件 里 面 输出 : 
<?php echo $name.[.$email.".$phone.];?> 
如 果 采 用 内 置 的 模板 引擎 ， 可 以 使 用 : 


{$name} [{$email} {$phonej] 





输出 同样 的 内 容 。 
1.3 ”指定 模板 文件 
模板 定义 后 就 可 以 泻 染 模板 输出 ， 系 统 也 支持 直接 泻 染 内 容 输出 ， 模 板 赋值 必须 在 模板 泻 染 之 前 


操作 。 演 染 模板 输出 最 常用 的 是 使 用 display0 方 法 ， 调 用 格式 如 下 : 
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display( [模板 文件 ][, 字 符 编码 ]L, 输出 类 型 ]) 





模板 文件 的 写法 如 表 9.5 所 示 。 
表 9.5 display 模板 用 法 
用 法 描 述 
不 带 任何 参数 自动 定位 当前 操作 的 模板 文件 
[模块 @][ 控 制 器 :][ 操 作 ] 常用 写法 ， 支 持 跨 模 块 模板 主题 ， 可 以 和 theme0 方 法 配合 
完整 的 模板 文件 名 直接 使 用 完整 的 模板 文件 名 (包括 模板 后 缀 ) 





下 面 是 一 个 最 典型 的 用 法 ， 不 带 任何 参数 : 


// 不 带 任何 参数 ， 自 动 定位 当前 操作 的 模板 文件 
S$this->display(); 


表示 系统 会 按照 默认 规则 自动 定位 模板 文件 ， 其 规则 是 : 
如 果 当 前 没有 启用 模板 主题 则 定位 到 : 


当前 模块 /默认 视图 目录 /当前 控制 器 /当前 操作 .html 





如 果 有 启用 模板 主题 则 定位 到 : 
当前 模块 /默认 视图 目录 /当前 主题 /当前 控制 器 /当前 操作 .html 
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如 果 有 更 改 TMPL FILE_DEPR 设置 (假设 TMPL FILE_DEPR'=>'') 的 话 ， 则 上 面 的 自动 定位 规 
则 变 成 : 

当前 模块 / 蒜 认 视图 目录 /当前 控制 器 当前 操作 html 

和 

当前 模块 / 蒜 认 视图 目录 /当前 主题 /当前 控制 器 _ 当 前 操 fhm 





所 以 通常 display0 方 法 无 须 带 任何 参数 即 可 输出 对 应 的 模板 ， 这 是 模板 输出 的 最 简单 的 用 法 。 
人 

说 明 通常 默认 的 视图 目录 是 View。 
如 果 没 有 按照 模板 定义 规则 来 定义 模板 文件 或 者 需要 调用 其 他 控制 器 下 面 的 某 个 模板 )， 可 以 


使 用 : 


// 指 定 模板 输出 
Sthis->display('edit'); 


表示 调用 当前 控制 器 下 面 的 edit 模板 : 





Sthis->display('Member:read'); 





表示 调用 Member 控制 器 下 面 的 read 模板 。 
9.12 内置 ThinkTemplate 模板 引擎 


9.12.1 变量 输出 


在 模板 中 输出 变量 的 方法 很 简单 ， 例 如 ， 在 控制 器 中 给 模板 变量 赋值 的 代码 如 下 : 


$name = 'ThinkPHP"; 
Sthis->assign('name', $name); 
Sthis->display(); 


然后 就 可 以 在 模板 中 使 用 : 


Hello,{$name}! 


模板 编译 后 的 结果 就 是 : 


Hello,<?php echo($name);?>! 
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这 样 ， 运 行 的 时 候 就 会 在 模板 中 显示 : 
Hello,ThinkPHP! 


注意 模板 标签 的 {和 $ 之 间 不 能 有 任何 的 空格 ,否则 标签 无 效 ， 将 不 会 正常 输出 name 变量 , 而 是 直 
接 保持 不 变 输出 ， 输 出 结果 如 下 : 





Hello,{$name}! 
普通 标签 默认 开始 标记 是 {, 结束 标记 是 }。 也 可 以 通过 设置 TMPL L DELIM 和 TMPL R_DELIM 
进行 更 改 。 例 如 ， 在 项 目 配 置 文 件 中 定义 : 


TMPL_L_DELIM'=>'<f， 
TMPL_R_DELIM'=>")>', 


那么 ， 上 面 的 变量 输出 标签 就 应 该 改 成 : 


Hello,<{$name}>! 


后 面 的 内 容 都 以 默认 的 标签 定义 来 说 明 。 

模板 标签 的 变量 输出 根据 变量 类 型 有 所 区 别 ， 刚 才 输 出 的 是 字符 串 变量 ， 如 果 是 数组 变量 ， 例 如 : 
$data[name'] = 'ThinkPHP"; 

$data[email] = thinkphp@qq.com' 

Sthis->assign('data', $data); 


那么 ， 在 模板 中 可 以 用 下 面 的 方式 输出 : 


Name: {$data.name} 
Email: {$data.email} 


或 者 用 下 面 的 方式 也 有 效 : 


Name: {$data[name]} 
Email: {$data[email]} 











SO 
“” 当 输出 多 维 数组 时 ， 往 往 要 采用 后 面 一 种 方式 。 
如 果 data 变量 是 一 个 对 象 并 且 包 含有 name 和 email 两 个 属性 ) ， 那 么 可 以 用 下 面 的 方式 输出 : 


Name: {$data:name} 
Email: {$data:email} 


或 者 





Name: {$data->name} 
Email: {$data->email} 


© 
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9.12.2 ”使 用 函数 


如 果 需 要 对 模板 中 变量 使 用 函数 ， 例 如 ， 对 模板 中 输出 变量 使 用 md5 加 密 ， 则 可 以 使 用 如 下 代码 : 


{$data.name|md5} 





编译 后 的 结果 是 : 


如 果 函 数 有 多 个 参数 需要 调用 ， 则 可 以 使 用 如 下 代码 : 


{$create_time|date="y-m-d",## 机 


上 述 代 码 表示 date 函数 传 入 两 个 参数 ， 每 个 参数 用 去 号 分 割 ， 这 里 第 一 个 参数 是 y-m-d， 第 二 个 
参数 是 前 面 要 输出 的 create_time 变量 ， 因 为 该 变量 是 第 二 个 参数 ， 因 此 需要 用 故 # 标 识 变量 位 置 ， 编 译 


后 的 结果 是 : 





<?php echo (date("y-m-d",$create_time)); ?> 





9.12.3 ”内 置 标签 


变量 输出 使 用 普通 标签 就 足够 了 ， 但 是 要 完成 其 他 的 控制 、 循 环 和 判断 功能 ， 就 需要 借助 模板 引擎 的 
标签 库 功能 ， 系 统 内 置 标签 库 的 所 有 标签 无 须 引入 标签 库 即 可 直接 使 用 。 常 用 内 置 标签 如 表 9.6 所 示 。 


表 9.6 ThinkPHP 内 置 标签 


























标 签 名 作 用 包含 属性 
include 包含 外 部 模板 文件 (闭合 ) file 
volist 循环 数组 数据 输出 name.id,offset,length,key.mod 
foreach 数组 或 对 象 遍 历 输出 name,item.key 
for For 循环 数据 输出 name.from.to.before.step 
switch 分 支 判 断 输出 name 
Case 分 支 判断 输出 〈 必 须 和 switch 配套 使 用 value.break 
compare 比较 输出 (包括 eq、neq、lt、gt、egt、clt、heq、nheq 等 别名 ) | name.value.type 
a 判断 数据 是 否 为 空 name, 
assi! 变量 赋值 (闭合 ) name.value 
让 条 件 判 断 输 出 condition 
9.12.4 “模板 继承 


模板 继承 是 一 项 更 加 灵活 的 模板 布局 方式 ， 模 板 继承 不 同 于 模板 布局 ， 甚 至 来 说 ， 应 该 在 模板 布 


PHP 项 目 开发 全 程 实 录 (第 4 版 ) 


局 的 上 层 。 模 板 继承 其 实 并 不 难 理解 ， 就 好 比 类 的 继承 一 样 ， 模 板 也 可 以 定义 一 个 基础 模板 〈 或 者 是 
布局 ) ， 并 且 其 中 定义 相关 的 区 块 (block) ， 然 后 继承 〈extend) 该 基础 模板 的 子 模板 中 就 可 以 对 基 
础 模板 中 定义 的 区 块 进行 重 载 。 

因此 ， 模 板 继承 的 优势 其 实 是 设计 基础 模板 中 的 区 块 (block) 和 子 模板 中 替换 这 些 区 块 。 每 个 
块 由 <block></block> 标 签 组 成 。 下 面 就 是 基础 模板 中 的 一 个 典型 的 区 块 设计 《用 于 设计 网 站 标题 〉: 


<block name="title"><title> 网 站 标题 </title></block> 








Kl 





block 标签 必须 指定 name 属性 来 标识 当前 区 块 的 名 称 , 这 个 标识 在 当前 模板 中 应 该 是 唯一 的 , block 
标签 中 可 以 包含 任何 模板 内 容 ， 包 括 其 他 标签 和 变量 ， 例 如 : 


<block name="title"><title>{$web_title}</title></block> 





甚至 还 可 以 在 区 块 中 加 载 外 部 文件 : 


<block name="include"><include file="Public:header" /></block> 


9.13 本 章 总 结 


本 章 运用 软件 工程 思想 中 最 流行 的 MVC 设计 理念 ， 通 过 一 个 业界 比较 知名 的 国产 框架 编写 而 成 。 
通过 对 本 章 的 学 习 ， 读 者 可 以 了 解 PHP 网 站 程序 的 开发 流程 ， 并 且 了 解 ThinkPHP 框架 开发 的 具体 事 
宜 。 希 望 对 读者 日 后 的 程序 开发 有 所 帮助 。 
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随 着 20 世纪 PC 机 【个 人 计算 机 ) 的 发 展 和 互联 网 的 普及 ， 电 子 
商务 从 报 文 时 代 进 入 到 了 Internet 时 代 , 并 逐渐 被 大 众 所 了 解 和 接受 。 
电子 商务 (Electronic Commerce，EC) 是 目前 发 展 较 快 的 一 种 商务 模 
式 。 迄 今 为 止 ， 不 同 领 域 的 人 对 EC 的 理解 各 有 不 同 。 简 单 地 说 ，EC 
是 一 种 基于 Internet， 利 用 计算 机 硬件 、 软 件 等 现 有 设备 和 协议 进行 
各 种 商务 活动 的 方式 。 本 章 应 用 ThinkPHP 3.2.3 版 本 实现 一 个 仿 京东 
的 B2C (Business-to-Customer) 商城 一 一 51 购 商 城 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 


Lad 
Lad 
[ed 
Lad 
[ed 
Lad 
Lad 
Lad 


了 解 如 何 进 行 系统 分 析 

了 解数 据 库 设计 流程 
熟悉 搭建 系统 架构 的 方法 
掌握 ThinkPHP 技术 的 应 用 
掌握 注册 登录 的 实现 方法 
掌握 多 级 分 类 的 实现 方法 
党 所 购物 车 的 实现 方法 
掌握 订单 的 处 理 方法 





配置 说 明 
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10.1 开发 背景 





自 20 世纪 90 年 代 ， 互 联网 蓬勃 发 展 ， 为 企业 提供 了 一 个 全 新 的 机 遇 。 企 业 网 站 、 电 子 商 务 成 为 
热门 话题 。 其 中 ， 电 子 商 务 更 是 关系 到 经 济 结构 、 产 业 升 级 和 国家 整体 经 济 竞争 力 。 为 此 ， 我 国 已 经 
将 发 展 电子 商务 列 为 信息 化 建设 的 重要 内 容 ， 并 努力 创造 条 件 ， 积 极地 推进 电子 商务 的 发 展 。 

据 美 国 在 线 (AOL) 和 Henley Centre 联合 进行 的 一 项 调查 显示 : 国外 有 80% 的 受 调查 者 会 选择 网 
上 购物 或 寻求 帮助 ，10% 的 受 调查 者 会 选择 熟悉 的 品牌 或 厂商 来 购买 。 而 在 国内 ， 自 1997 年 拉 开 了 电 
子 商 务 的 序幕 ， 几 乎 每 天 都 有 新 的 网 站 诞生 ， 厂 商 所 在 地 也 从 上 海 、 广 州 、 深 圳 等 沿海 发 达 地 区 扩展 
到 全 国 各 大 、 中 城市 。 











10.2 需求 分 析 


随 着 “地 球 村 ”概念 的 兴起 ， 网 络 已 经 深入 到 人 们 生活 的 每 一 个 角落 。 世 界 越 来 越 小 ， 信 息 的 传 
播 越 来 越 快 ， 内 容 也 越 来 越 丰富 。 现 在 ， 人 们 对 于 在 网 络 上 寻求 信息 和 服务 已 不 再 满足 于 简单 的 信息 
获取 上 ， 人 们 更 多 的 是 需要 在 网 上 实现 方便 的 、 便 捷 的 、 可 交互 式 的 网 络 服务 。 电 子 商 务 则 正好 满足 
了 人 们 的 需求 。 它 可 以 让 人 们 在 网 上 实现 互动 的 交流 及 足 不 出 户 地 购买 产品 ， 向 企业 发 表 自己 的 意见 、 
服务 需求 及 有 关 投 诉 ， 并 且 通 过 网 站 的 交互 式 操作 向 企业 进行 产品 的 咨询 、 得 到 相应 的 回馈 及 技术 支 
持 。 精 明 的 商家 绝 不 会 错过 这 样 庞 大 的 市 场 ， 越 来 越 多 的 企业 已 经 开展 了 电子 商务 活动 。 加 入 电子 商 
务 的 行列 也 许 不 会 让 企业 马上 见 到 效益 ， 但 不 加 入 则 一 定 会 被 时 代 所 抛弃 。 


10.3 系统 分 析 


10.3.1 系统 目标 


根据 客户 提供 的 需求 和 对 实际 情况 的 考察 与 分 析 ， 该 电子 商务 应 该 具备 如 下 特点 : 

首页 设计 要 能 够 吸引 用 户 的 目光 ， 整 个 页 面 要 以 简洁 为 主 ， 突 出 重点 。 

可 操作 性 强 ， 避 免 复 杂 的 、 有 异议 的 链接 。 

浏览 速度 快 ， 尽 量 避 免 长 时 间 打 不 开 页 面 的 情况 发 生 。 

商品 信息 部 分 有 实物 图 例 ， 图 像 清楚 、 文 字 醒 目 。 

详细 的 商品 查询 功能 ， 可 以 通过 商品 的 各 个 属性 来 搜索 。 

详细 的 流程 介绍 ， 从 浏览 商品 到 购买 结账 ， 各 个 步骤 之 间 的 联系 最 好 能 以 图 例 来 说 明 。 
提供 在 线 咨询 。 

后 台 可 以 对 用 户 信息 和 商品 信息 进行 详尽 的 查看 和 管理 。 


办 办 办 办 办 办 办 
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回 ”完善 的 订单 管理 。 
回 ” 易 维护 ， 并 提供 二 次 开发 支持 。 


10.3.2 ”系统 功能 结构 


51 购 商城 分 前 台 系 统 和 后 台 系 统 。 下 面 分 别 给 出 前 、 后 台 的 系统 功能 结构 图 。51 购 商 城 前 台 系统 
功能 结构 如 图 10.1 所 示 。 


| 5 购 商城 前 台 
| I | I | 
| 用户 中 心 订单 中 心 商品 展示 购物 车 | | 文章 展示 | 
| 1 
处 | | 二 | 村 | 

用 | 用 || 改 | | 我 褒 | | 章 || 菏 || 意 | | 加 | | 突 | | 文 文 
户 | 户 | 个 | 的 | 的 || 品 | | 品 || 门 || 品 | | 旭 || 贻 || 次 | 章 章 
注 || 登 || 人 | | 订 || 收 || 评 | | 分 || 商 || 详 | | 物 || 物 || 物 | 列 详 
册 | | 水 || 信 | 价 | | 类 || 品 | 情 | | 车 || 车 || 车 | | 表 情 


图 10.1 51 购 商城 前 台 系统 功能 结构 图 
51 购 商 城 后 台 系 统 功能 结构 图 如 图 10.2 所 示 。 











| 。 51 购 商城 后 台 
[ | 1 
商品 管理 会 员 管理 订单 管理 | | ”广告 管理 
hs BA 、 [ ] 
| | 
| 
商 | | 商 | | 商 | | 商 || 商 || 商 会 | | 会 | | 订 || 订 || 发 | 文 | 文 
品 || 品 | | 品 | | 品 || 品 | | 品 员 | | 员 | | 单 || 单 || 货 | | 章 章 | 
分 || 列 | | 类 | | 规 || 属 | | 评 列 | | 等 | | 列 || 详 || 设 | 分 列 
类 || 表 || 型 | | 格 || 性 || 论 表 | 级 崩 | | 置 | | 类 表 





10.2 51 购 商 城 后 台 系 统 功能 结构 图 
10.3.3 ”功能 预览 


51 购 商城 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 几 个 典型 
功能 的 页 面 ， 其 他 页 面 参见 资源 包 中 的 源 程序 。 
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51 购 商 城 购 网 站 主页 如 图 10.3 所 示 ， 该 页 面 展 示 网 站 商品 分 类 、 热 门 商品 、 推 荐 商品 以 及 网 站 的 
最 新 公告 和 会 员 登 录 窗 口 。 商 品 列表 展示 页 面 如 图 10.4 所 示 ， 该 页 面 分 页 展示 网 站 的 所 有 商品 。 商 品 详 
情 页 如 图 10.5 所 示 ， 该 页 面 展示 商品 的 详细 信息 。 购 物 车 页 面 如 图 10.6 所 示 ， 该 页 面 展 示 会 员 在 本 站 
购买 的 商品 。 
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10.3.4 系统 流程 图 


51 购 商 城 涉及 很 多 业务 流程 ， 其 中 最 重要 
的 就 是 用 户 购物 流程 。 该 流程 包括 用 户 登录 、 用 户 





选择 商品 、 加 入 购物 车 、 结 算 订单 等 ， 具 体 流 
程 如 图 10.7 所 示 。 EE 
-一 | 商城 主页 面 
10.3.5 ”开发 环境 选 购 商品 
在 开发 51 购 商城 时 ， 该 项 目 使 用 的 软件 开 。 是 加 入 购物 车 
发 环境 如 下 。 T 
加 ”操作 系统 ，Windows 7 及 以 上 /Linux。 
加 ”集成 开发 环境 : phpStudy。 一 能 续 购 物 
回 PHP 版 本 : PHP7。 
回 MySQL 图 形 化 管理 软件 : Navicat for ! 
MySQL。 去 结算 
回 开发 工具 : PhpStorm 9.0。 几 
回 ThinkPHP 版 本 : 3.2.3。 是 瑾 登录 “> 盏 | 用 户 登 录 
回 浏览 器 : 谷歌 浏览 器 。 T 
10.3.6 ”文件 夹 组 织 结构 去 支 什 上 
2 
在 进行 网 站 开发 前 ， 首 先 要 规划 网 站 的 架 退出 系统 
构 。 也 就 是 说 ， 建 立 多 个 文件 夹 对 各 个 功能 模 
块 进行 划分 ， 实 现 统一 管理 ， 这 样 做 易于 网 站 图 107 购物 流程 图 


的 开发 、 管 理 和 维护 。 本 项 目 中 ， 使 用 默认 的 
ThinkPHP 目录 结构 ， 将 Home 文件 夹 作为 前 台 模 块 ，Admin 文件 夹 作为 后 台 模块 ， 如 图 10.8 所 示 。 











7Y 癌 Application 一 一 一 一 一 一 一 一 一 一 一 一 应 用 目录 
* 疡 Admn 一 一 一 一 一 一 一 一 一 一 一 后 台 模 块 
站 common 一 一 一 一 一 一 一 一 一 一 一 通用 模块 
PDDHome 前 台 模块 
PORuntime 缓存 文件 

» DPpublic 一 一 一 一 一 一 一 一 一 一 一 一 公共 文件 

* 站 Thinkphp 一 一 一 一 一 一 一 一 一 一 一 框架 核心 文件 
八 adminphp 一 一 一 一 一 一 一 一 一 一 一 一 后 人 台 入 口 文件 
要 ndexphp 一 一 一 一 一 一 一 一 一 一 一 前 台 入 口 文件 
目 README.md 一 一 一 一 一 一 一 一 一 一 一 一 使 用 说 明 





图 10.8 文件 夹 组 织 结构 
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10.4 数据 库 设 计 





Ss 
无 论 是 什么 系统 软件 ， 其 最 根本 的 功能 就 是 对 数据 的 操作 与 使 用 。 所 以 ， 一 定 要 先 做 好 数据 的 分 
析 、 设 计 与 实现 ， 然 后 再 实现 对 应 的 功能 模块 。 


10.4.1 数据 库 分 析 


本 系统 采用 MySQL 作为 数据 库 ， 数 据 库 名 称 为 shop， 其 数据 表 名 称 及 作用 如 表 10.1 所 示 。 
表 10.1 数据 库 表 结 构 
































表 名 含义 作 用 
ad 广告 表 用 于 存储 广告 信息 
ad _position 广告 位 置 表 用 于 存储 广告 分 布 的 位 置信 息 
admin 管理 员 表 用 于 存储 管理 员 用 户 信息 
article 文章 表 用 于 存储 商城 中 的 文章 信息 
article_cat 文章 分 类 表 用 于 存储 文章 的 分 类 信息 
brand 品牌 表 用 于 存储 商品 品牌 信息 
cart 购物 车 表 用 于 存储 购物 车 中 信息 ， 包 括 未 登录 用 户 的 购物 车 信息 
comment 评论 表 用 于 存储 商品 评论 信息 
confip 网 站 配置 表 用 于 存储 网 站 配置 信息 
goods 商品 表 用 于 存储 商品 信息 
goods attr 属性 映射 表 用 于 存储 商品 和 属性 的 对 应 关系 信息 
goods attribute 商品 属性 表 用 于 存储 商品 属性 信息 
goods_catego 商品 分 类 表 用 于 存储 商品 分 类 信息 
goods_collect 商品 收藏 表 用 于 存储 用 户 收藏 的 商品 信息 
goods images 商品 图 片 表 用 于 存储 商品 图 片 信息 
goods type 商品 类 型 表 用 于 存储 商品 类 型 信息 
order 订单 表 用 于 存储 用 户 订单 信息 
order action 订单 操作 表 用 于 存储 订单 操作 信息 ， 包 括 下 单 、 取 消 订单 等 
order goods 订单 商品 表 用 于 存储 订单 商品 信息 
region 地 区 表 用 于 存储 地 区 信息 
spec 商品 规格 表 用 于 存储 商品 规格 信息 
spec_ goods price 规格 价钱 表 用 于 存储 商品 规格 对 应 的 价钱 信息 
spec_image 规格 图 片 表 用 于 存储 商品 规格 图 片 信息 
spec_item 规格 选项 表 用 于 存储 商品 规格 选项 信息 
user address 用 户 地 址 表 用 于 存储 用 户 地 址 信息 
user level 用 户 等 级 表 用 于 存储 用 户 等 级 信息 
Users 用 户 表 用 于 存储 用 户 信息 
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10.4.2 数据库 逻辑 设计 
1. 创建 数据 表 


由 于 篇 幅 所 限 ， 这 里 只 给 出 较 重 要 的 数据 表 的 部 分 字段 ， 完 整数 据 表 请 参见 本 书 附带 资源 包 。 


回 admin《〈 后 台 管 理 员 表 ) 
表 admin 用 于 保存 后 台 管理 员 数 据 信 息 ， 其 结构 如 表 10.2 所 示 。 


表 10.2 管理 员 表 结构 












































字 段 名 数据 类 型 备 注 
admin id smallint(5) unsigned 用 户 id 
user name varchar(60) 用 户 名 
email varchar(60) email 
password | varcha32) | | wo | 密码 
aldtime [nD | o。 | x | | 测 喇 则 
lastbgin | iD) | 0 | No | | 最 后 登录 时 间 
last_ip | NO | | 最 后 登录 
回 users〈 用 户 表 ) 

表 users 用 于 存储 用 户 数据 信息 ， 其 结构 如 表 10.3 所 示 。 
表 10.3 用 户 表 结构 
字 段 名 数据 类 型 自动 递增 备注 

user id mediumint(8) unsigned | | wo | 是 表 id 

email varchar(60) 邮件 

password varchar(32) 密码 

SeX tinyint(1) unsigned 0: 保密 ，1: 男 ，2: 女 

birthday int(11) 生日 

pay_points | int(10) unsigned 消费 积分 

address id | mediumint(8) unsigned 默认 收 货 地 址 

reg time int(10) unsigned 注册 时 间 

last_login int(11) unsigned 最 后 登录 时 间 

last ip varchar(15) 最 后 登录 ip 

qq varchar(20) QQ 

mobile varchar(20) 手机 号 码 

head pic varchar(255) 头像 

Province int(6) 省 份 

city int(6) 市 区 

district int(6) 县 

nickname Varchar(50) 第 三 方 返回 昵称 

level tinyint(1) 会 员 等 级 
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goods〈 商 品 表 ) 
表 goods 用 于 存储 商品 信息 ， 其 结构 如 表 10.4 所 示 。 


表 10.4 商品 表 结 构 




















字 段 名 数据 类 型 默 认 值 | 允许 为 空 | 自动 递增 备 注 
goods id mediumint(8) unsigned 是 商品 id 
cat id int(11) unsigned 0 分 类 id 
goods sn varchar(60) 商品 编号 
goods name varchar(120) 商品 名 称 
click count int(10) unsigned 0 单 击 数 
brand id smallint(5) unsigned 品牌 id 





0 NO 
store_count smallint(5) unsigned | 10 | 库存 数量 
comment count | smallint(5) | 0o | 商品 评论 数 
market_price decimal(10,2) unsigned | oo | No | 市 场 价 
shop_price decimal(10,2) unsigned | oo | 本 店 价 
cost price decimal(10.2) | oo | YEs | 商品 成 本 价 
keywords varchar(255) | | 商品 关键 词 
goods remark | varchar(255) | | 商品 简单 描述 
goods_content text | | 商品 详细 描述 
original img Varchar(255) | | | No | 商品 上 传 原始 
is_on sale tinyint(1) unsigned | 一 二 二 | | No | 是 否 上 架 


;4 
4 

















on _time int(10) unsigned | 0o | | No | 商品 上 架 时 间 
sort smallint(4) unsigned 商品 排序 
is_recommend tinyint(1) unsigned | 0o | 是 否 推荐 

is_ new tinyint(1) unsigned | 0 | 是 否 新 品 

is_hot tinyint(1) 0 YES 是 否 热 卖 
last_update int(10) unsigned 0 NO 最 后 更 新 时 间 
goods type smallint(5) unsigned 0 NO 商品 所 属 类 型 id 
spec_type smallint(5) 0 YES 商品 规格 类 型 id 


2. 数据 库 连接 相关 配置 
在 ThinkPHP 全 局 配置 文件 中 配置 数据 库 信 息 ， 具 体 配 置 代码 如 下 : 


倒 程 01 代码 位 置 : 资源 包 \TM\10\Shop\ApplicatiomNCommon\Conf\config.php 





<?php 
return array( 
/| 数据 库 配 置 
'DB_TYPE' => ,mysql'， /数据 库 类 型 
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'DB_HOST' => '127.0.0.1， /host 地 址 
'DB_USER' => 'root', /数据 库 用 户 名 
'DB_PWD' => root， /| 数据 库 密码 
'DB_NAME' => 'shop', // 数 据 库 名 称 


10.5 前台 首页 设计 





当 用 户 访问 51 购 商 城 时 ， 首 先进 入 的 便 是 前 台 首 页 。 前 台 首 页 是 对 整个 网 站 总 体内 容 的 概述 。 在 
51 购 商 城 的 前 台 首 页 中 ， 主 要 包含 以 下 内 容 。 
回 ”商品 分 类 模块 ， 主 要 包括 首页 “图 书 、 音 像 、 电 子 书 ” 等 一 级 分 类 、“ 科 技 、 音 像 ”等 二 级 分 
类 和 “计算 机 与 互联 网 、 建 筑 ” 等 三 级 分 类 ， 如 图 10.9 所 示 。 
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图 10.9 前 台 分 类 、 搜 索 、 导 航 、 幻 灯 片 模块 效果 图 


网 站 菜单 导航 模块 ， 主 要 包括 “首页 ”以 及 “计算 机 编程 ”“ 手 机 ”“ 平 板 电 脑 ” 等 热门 的 商 
品类 别 ， 如 图 10.9 所 示 。 

搜索 模块 : 主要 用 于 商品 信息 的 快速 搜索 ， 如 图 10.9 所 示 。 

幻灯 片 模块 ， 主 要 用 于 以 幻灯 片 的 方式 演示 商品 ， 如 图 10.9 所 示 。 

热卖 商品 模块 : 主要 展示 商城 重点 推荐 的 商品 及 详细 信息 查看 ， 如 图 10.10 所 示 。 

推荐 商品 模块 : 主要 用 于 展示 每 个 分 类 下 的 推荐 商品 ， 如 图 10.11 所 示 。 
文章 分 类 和 信息 模块 : 主要 显示 分 类 的 文章 , 为 用 户 提供 相应 的 帮助 和 支持 , 如 图 10.12 所 示 。 
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10.5.2 前台 首页 技术 分 析 


前 台 首 页 中 涉及 较 多 的 模块 ， 需 要 注意 模块 之 间 的 独立 性 和 共性 。 例 如 幻灯 片 和 广告 栏 ， 它 们 的 
数据 均 来 源 于 ad 广告 表 ， 只 是 因为 所 在 位 置 不 同 ， 所 以 在 首页 的 展现 形式 不 同 。 对 于 商品 分 类 ， 共 有 
3 级 分 类 ， 所 有 数据 均 来 源 于 goods_catogary 表 ， 根 据 level 字段 判断 当前 分 类 等 级 ， 根 据 parent id 来 
判断 所 属 上 级 分 类 。 在 前 台 首 页 中 ， 我 们 重点 介绍 商品 分 类 模块 和 推荐 商品 模块 。 


10.5.3 ”商品 分 类 模块 的 实现 过 程 


国 商品 分 类 模块 使 用 的 数据 表 : goods_category 
1. 获取 分 类 数据 


本 项 目 中 ， 不 只 在 前 台 首页 包含 商品 分 类 数据 ， 在 其 他 页 面 ， 当 鼠标 悬浮 在 “全 部 商品 ”内 容 上 
时 ， 也 会 显示 商品 分 类 数据 。 所 以 ,我 们 将 商品 分 类 作为 通用 数据 ， 写 在 BaseController.class.php 父 类 
文件 中 ， 然 后 令 相应 的 控制 器 类 通过 extends 继承 BaseController.class.php 父 类 。 关 键 代 码 如 下 : 


倒 程 02 代码 位 置 ， 资源 包 \TM\10\Shop\Application\Home\Controller\IndexController.class.php 
<?php 





class IndexController extends BaseController { 
public function index(){ 
// 省 略 其 余 代码 
} 


当 访 问 前 台 首页 ， 即 Home 模块 的 Index 控制 器 的 index0 方 法 时 ， 程 序 会 优先 执行 其 父 类 的 
_initialize0 初 始 化 方法 。 关 键 代码 如 下 : 


倒 程 03 代码 位 置 ， 资源 包 \TM\10\ShopWApplication\Home\Controller\BaseController.class.php 
<?php 








namespace Home\Controller; 
use Think\Controller; 


class BaseController extends Controller { 


六 
* 初始 化 操作 
最 
public function _initialize() { 
/省 略 其 余 代码 
S$this->public_assign(); // 调 用 public_assign() 方 法 
} 
er 
* 保存 公告 变量 到 smarty 模板 中 
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bd 
public function public_assign() 


1/ 省略 其 余 代码 
$goods_category_tree = get_goods_category_tree(); // 获 取 商 品 一 二 三 级 分 类 
$this->cateTrre = $goods_category_tree; // 分 类 赋值 


S$this->assign('goods_category_tree', $goods_category_tree); 。“ /模板 赋值 
} 


上 述 代码 中 ， 调 用 了 get_goods_category_tree0 函 数 ， 该 函数 主要 用 于 将 所 有 分 类 按照 一 、 二 、 三 
级 分 类 的 数据 格式 保存 在 数组 中 。 具 体 代码 如 下 : 


倒 程 04 代码 位 置 ， 资源 包 \TM\10\Shop\Application\Home\Common\function.php 
Jr 
* 获取 商品 一 二 三 级 分 类 
* @return type 
本 
function get_goods_category_tree(){ 
$result = array(); 
/查询 所 有 显示 的 分 类 
$cat_list = M('goods_category')->where("is_show = 1")->order('sort_order')->select(); 
foreach ($cat_list as $val}{ 





if($vall'level] == 2X{ // 如 果 是 二 级 分 类 
$arr[$val[parent_id]i0 = $val; // 将 该 分 类 赋值 给 $arr 数组 的 元 素 
} 
if($vall'level] == 3X{ 1/ 如果 是 三 级 分 类 
$crr[f$val[parent_id]0 = $val; // 将 该 分 类 赋值 给 $crr 数组 的 元 素 
} 
if($valllevel] == 1X{ // 如 果 是 一 级 分 类 
S$tree[] = $val; // 将 该 分 类 赋值 给 $tree 数组 的 元 素 


} 
} 


// 人 遍历 二 级 分 类 ， 将 三 级 分 类 的 内 容 赋 值 给 二 级 分 类 
foreach ($arr as $k=>$vX{ 
foreach ($v as $kk=>$vvX{ 
// 三 级 分 类 内 容 作 为 其 二 级 分 类 的 sub_menu 
$arr[$kl[$kkl[sub_menu] = empty($crr[$vv['id"]]) ? array() : $crr[$vv[id]]; 


} 


// 人 遍历 一 级 分 类 ， 将 二 级 分 类 的 内 容 赋 值 给 一 级 分 类 

foreach ($tree as $val){ 
// 二 级 分 类 内 容 作 为 其 一 级 分 类 的 tmenu 
S$val[tmenu'] = empty($arr[$val[id]]) ? array() : $arr[$val['id"]]; 
$result[$val[id]] = $val; 


IE 
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return $result; 


} 
get_goods_category_tree0 函 数 返 回 值 示例 数据 如 下 : 


Array( 
[11] => Array( 
[id] => 11 
[name] => 图 书 、 音 像 、 电 子 书 
[parent_id] => 0 
[parent_id_path] => 0_11 
[levell => 1 
[tmenu] => Array( 
[0] => Array( 
[id] => 94 
[name] => 科技 
[parent_id] => 11 
[parent_id_path] => 0_11_94 
[levell => 2 
[sub_menu] => Array( 
[0] => Array( 
[id] => 842 
[name] => 计算 机 与 互联 网 
[parent_id] => 94 
[parent_id_path] => 0_11_94_842 
[level] => 3 
) 
[1] => Array( 
[id] => 836 
[Iname] => 建筑 
[parent_id] => 94 
[parent_id_path] => 0_11_94_836 
[levell => 3 
) 


从 上 面 示例 中 可 以 看 出 , get_goods_category_tree0 函 数 返 回 值 为 多 维 数 组 , 其 中 ID 为 11 的 记录 是 
一 级 分 类 ， 分 类 名 称 是 “图 书 、 音 像 、 电 子 书 ”。ID 为 94 的 记录 是 一 级 分 类 下 的 二 级 分 类 ， 分 类 名 
称 为 “科技 ”。ID 为 842 和 836 的 记录 为 二 级 分 类 下 的 三 级 分 类 ,分 类 名 称 分 别 是 “计算 机 与 互联 网 ” 
和 “建筑 ”。 

2. 泻 染 模板 

获取 完 分 类 数据 后 ， 接 下 来 就 需要 泻 染 模板 显示 数据 。 由 于 分 类 数据 是 一 个 多 维 数组 ， 所 以 使 用 
<foreach> 标 签 循 环 嵌 套 获取 各 级 分 类 数据 即 可 。 关 键 代码 如 下 : 


倒 竹 05 代码 位 置 资源 包 \TM\10\Shop\Application\Home\View\Public\header.html 
<!-- 遍 历 一 、 二 、 三 级 分 类 -> 
<foreach name="goods_category_tree" key="k" item='V'> 
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<if condition="$v[level] eq 1"> <!--if 标 签 判 断 是 否 为 一 级 分 类 --> 
<liclass="list-li"> 
<div class="list_a"> 
< 上 -输出 一 级 分 类 内 容 --> 
<h3><a href="{:U('Home/Goods/goodsList',array(id'=>$v[id"]))}"> 
<span>{$v[name]j</span> 
</a></h3> 
<p> 
<!- 选 出 3 个 二 级 标题 -> 
<assign name="index" value="1" /> 
<foreach name="v[tmenuy']" item="V2" key="k2" > 
<if condition="$v2[ parent_id'] eq $v['id]"> 
<?php if($index++ > 3) break; ?> 
<a href="{:U('Home/Goods/goodsList',array('id'=>$v2['id"]))}"> 
{$v2['name']} 
</a> 
</if> 
</foreach> 
</p> 
</div> 
<div class="list_b"> 
<div class="list_bigfl"> 
<!- 选 出 6 个 二 级 标题 --> 
<assign name="index" value="1" /> 
<foreach name="v[tmenu]" item="v2" key="k2" > 
<if condition="$v2[parent_id] eq Sv[id]"> 
<?php if($index++ > 6) break; ?> 
<a class="list_big_o ma-le-30" 
href="{:U('Home/Goods/goodsList',array('id'=>$v2['id"]))}"> 
{tv2[name] <i>></i> 
</a> 
<iif> 
</foreach> 
</div> 
<div class="subitems"> 
<!- 遍 历 二 级 标题 -> 
<foreach name="v[tmenu]" item="v2" key="k2" > 
<if condition="$v2[parent_id] eq $v[id]"> 
<dl class="ma-to-20 cl-bo"> 
<dt class="bigheader wh-sp"> 
<a href="{:U('Home/Goods/goodsList', 
array(id'"=>$v2[id"]))}"> {$v2['name'"]} 
</a><i>></i> 
</dt> 
<dd class="ma-le-100"> 
< 上 -遍历 二 级 标题 下 的 三 级 标题 -> 
<foreach name="v2[sub_ menu] item="v3" key="k3" > 
<if condition="$v3[parent_id] eq $v2[id]> 





全 
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<a class="hover-r ma-le-10" 
href="{:U(Home/Goods/goodsList', 
array(id=>$v3[id]))"> {$v3[name']} 


</a> 
</if> 
</foreach> 
</dd> 
</dl> 
</if> 
</foreach> 
</div> 
</div> 
</li> 
</if> 
</foreach> 





运行 结果 如 图 10.13 所 示 。 


计算 机 编程 


+ 和 WS 于 风 二 拒 。 工 业 x 
家 用 电 湛 


电脑 办 公 
男装 女装、 便装、 内 胡 
坪 要 用 品 、 儿 童 玩具 


个 人 化 站 > = ze 吾 。 固 囊 粮 
美食 时兴 美 站 。 至 属 。 于 工 0 。 字 才 所 再 儿 。 两 性。 孕 产 体育 。 健 且 保健 交融 / 世 图 


WW、 驱 的 、 通 信 














10.13 三 级 分 类 效果 图 
10.5.4 推荐 商品 模块 的 实现 过 程 


国 推荐 商品 模块 使 用 的 数据 表 : goods、goods_category 

由 于 商品 数量 较 多 ， 在 前 台 首 页 只 能 展示 一 部 分 商品 数据 ， 所 以 从 goods 商品 表 中 筛选 数据 时 ， 
只 筛选 出 满足 以 下 条 件 的 数据 : 

回 goods_category 表 商 品 分 类 is_show 字段 值 为 1 的 分 类 ， 且 最 多 只 筛选 7 个 分 类 。 

回 goods 表 is_on_sale 字段 值 为 1， 即 在 售 的 商品 ， 且 最 多 只 筛选 7 个 商品 。 

具体 实现 方式 代码 如 下 : 


倒 程 06 代码 位 置 资源 包 \TM\10\Shop\Application\Home\Controller\IndexController.class.php 





$category1 = M('goods_category')->where(array('is_show'=>1,'level'=>1)) 
->limit(7)->select(); /筛选 一 级 分 类 


439 
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foreach($category1 as $key=>$v ){ 
$category2 = M('goods_category')->where(array('is_show=>1,'parent_id'=>$v['id"])) 
->field('id,name')->select(); ”// 筛 选 二 级 分 类 
$category[$v[name]l[sub_category] = $category2; 


$cat id_arr = getCatGrandson($v['id]); /找到 一 级 下 面 的 所 有 子 分 类 id 
S$sub_id_str = implode(',',$cat_id_arn); // 将 子 分 类 id 拼接 成 字符 串 
$map['cat_id] = array('in', $sub_id_str); /搜索 条 件 :; 商品 分 类 id 在 子 类 id 中 
$map[is_on_sale] = 1; /搜索 条 件 ， 商品 在 售 


// 从 商品 表 中 ， 筛 选 7 条 满足 以 上 2 个 条 件 的 记录 

$category[$v[name]l[goods] = M('goods')->where($map)->limit(7) 
->field('goods_id,goods_name,keywords,goods_remark,shop_price') 
->order('goods_id')->select(); 


} 
Sthis->assign('category", $category); /模板 赋值 





运行 结果 如 图 10.14 所 示 。 
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图 10.14 ”推荐 商品 模块 






10.6 登录 模块 设计 


ks 
10.6.1 登录 模块 概述 


用 户 登 录 模块 是 会 员 功 能 的 窗口 。 匿 名 用 户 虽 然 也 可 以 访问 本 网 站 ， 但 只 能 进行 浏览 、 查 询 等 简 
单 操作 ， 而 会 员 则 可 以 购买 商品 、 查 看 订单 、 添 加 收 货 地 址 等 。 登 录 模 块 包 括 用 户 注册 和 用 户 登 录 。 


全 
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10.6.2 ”登录 模块 技术 分 析 


通过 使 用 Session 来 判断 用 户 是 否 登录 ， 如 果 Session 值 存在 ， 说 明 已 经 登录 ， 否 则 说 明 用 户 没有 
登录 ， 跳 转 至 登录 页 面 。 在 登录 页 面 填写 正确 的 用 户 名 和 密码 后 ， 登 录 成 功 , 将 用 户 信息 写 入 Session 。 
如 果 没 有 账号 ， 需 要 在 注册 页 面 注 册 一 个 新 用 户 。 


10.6.3 ”用 户 注册 功能 的 实现 过 程 


国 ”用户 注 册 功能 使 用 的 数据 表 : users 

会 员 注册 模块 主要 用 于 实现 新 用 户 注 册 成 为 网 站 会 员 的 功能 。 在 会 员 注册 页 面 中 ， 用 户 需 要 填写 
会 员 信息 ， 并 且 需 要 选中 “同意 《账号 服务 条 款 、 隐 私 政策 》” 选 项 ， 单 击 “ 注 册 ” 按 钮 ， 程 序 将 验 
证 输入 的 账户 是 否 唯一 。 如 果 唯 一 ， 就 把 填写 的 会 员 信息 保存 到 数据 库 中 ， 否 则 给 出 错误 提示 ， 需 要 
修改 唯一 后 ， 方 可 完成 注册 。 

会 员 注册 页 面 中 ， 主 要 包括 Form 表单 的 提交 和 表单 数据 的 验证 。Form 表单 主要 代码 如 下 : 


倒 程 07 代码 位 置 ， 资源 包 \TM\10\Shop\Application\Home\View\User\register.html 


<form id="register"> 

<div class="user-email"> 

<label for="email"><i class="mr-icon-envelope-o0"></i></label> 

<input type="email" name="email" id="email" placeholder=" 请 输入 邮箱 "> 
</div> 
<div class="user-pass"> 

<label for="password"><i class="mr-icon-lock"></i></label> 

<input type="password" name=" id="password" placeholder=" 设 置 密码 "> 
</div> 
<div class="user-pass"> 

<label for="password2"><i class="mr-icon-lock"></i></label> 

<input type="password" name="" id="password2" placeholder=" 确 认 密码 "> 
</div> 





<div class="user-pass"> 

<label for="mobile"><i class="mr-icon-mobile"></i></label> 

<input type="text" name="mobile" id="mobile" placeholder=" 请 输入 手机 号 "> 
</div> 
<div class="mr-cf"> 

<input type="button” value=" 注 册 " onClick="checkSubmit()" 

class="mr-btn mr-btn-primary mr-btn-sm mr-fl"> 
</div> 
</form> 





会 员 注册 页 面 的 运行 结果 如 图 10.15 所 示 。 
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10.15 ”会员 注册 页 面 运行 效果 


在 用 户 注册 时 ， 必 须 同意 “服务 条 款 ”， 否 则 提示 错误 信息 。 该 验证 功能 是 通过 JavaScript 代码 实 
现 的 。 当 用 户 单 击 “ 同 意 《 账 号 服务 条 款 、 隐 私 政策 》” 超 链接 时 ， 页 面 会 弹出 “服务 条 款 ” 的 具体 
内 容 。 该 功能 是 通过 Layerjs( 弹 层 插件 ) 实现 的 。 关 键 代 码 如 下 : 


倒 程 08 ”代码 位 置 ; 资源 包 \TM\10\Shop\Application\Home\View\User\register.html 
<script> 
// 单 击 提交 
function checkSubmit(){ 
/省略 其 余 代 码 
var agree = $('input[type="checkbox"]:checked').val(); /获取 账号 服务 条 款 
/检测 是 否 勾 选 注册 协议 
if(lagree}{ 
showErrorMsg(' 您 没有 同意 注册 协议 !); 
return false; 


} 
/ajax 异步 提交 到 后 台 验 证 
S$.ajax({ 
type : 'post', 
url : "{U('Home/User/register)}", 
data : {email:email,password:password,mobile:mobile}, 
dataType : json', 
success : function(resf 
if(res.status == 1){ 
layer.msg(res.msg,{icon:1,time:2000},function(X{ 
window.location.href = "{:-U('Home/Index/index")}"; // 跳 转 到 首页 
入 
jelse{ 
showErrorMsg(res.msg); // 显 示 错 误 信 息 
} 
上 
error : function(XMLHttpRequest, textStatus, errorThrown) { 
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showErrorMsg( 网 络 失败 ， 请 刷新 页 面 后 重 试 ); 


) 


} 

// 显 示 错 误 信息 的 方法 

function showErrorMsg(msgX{ 
layer.msg(msg,{icon:2,time:2000}); 


} 

// 显 示 协 议 内 容 

function showProtocol(){ 

var protocol = '<p style="padding: 10px">&nbsp &nbsp &nbsp 欢迎 来 到 明日 学 院 ， 为 了 

保障 您 的 权益 ， 请 在 使 用 明日 学 院 服务 之 前 ， 详 细 阅 读 此 服务 协议 (以 下 简称 “本 协议 ") 所 有 
内 容 ， 如 您 不 同意 本 协议 任何 条 款 ， 请 勿 注册 账号 或 使 用 本 平台 。 本 协议 内 容 包括 协议 正文 、 
本 协议 下 述 协议 明确 援引 的 其 他 协议 、 明 日 科技 公司 已 经 发 布 的 或 将 来 可 能 发 布 的 各 类 规则 。 
所 有 规则 为 本 协议 不 可 分 割 的 组 成 部 分 ， 与 协议 正文 具有 同等 法 律 效力 。 除 另行 明确 声明 外 ， 


您 使 用 明日 学 院 服务 均 受 本 协议 约束 。</p>"; 


layer.open({ 
title:' 协 议 内 容 '， // 弹 层 标题 
type: 1, // 弹 层 类 型 
skin: ‘ayui-layer-rim’, /加 上 边框 
area: [520px', '240px], // 弹 层 宽 和 高 
content: protocol // 弹 层 内 容 
入 
</script> 
运行 效果 如 图 10.16 所 示 。 


ed x 


欢迎 来 到 明日 字 院 ， 为 了 保 建 您 的 权益 ， 请 在 使 用 明日 他 院 芝 务 


之 前 ， 许 细 陪 读 此 夫 议 (也 攻 ” ) 所 有 内 容 ， 如 您 不 
和 


同志 李 协议 任何 条 款 包括 


明日 学 院 服务 均 受 下 协议 约束 。 





10.16 弹出 服务 协议 内 容 


从 上 述 代 码 中 可 以 看 出 ， 当 满足 验证 条 件 后 ， 单 击 “ 注 册 ” 按 钮 ， 会 通过 Ajax 异步 提交 的 方式 ， 
将 Form 表单 中 的 内 容 提交 到 Home 模块 下 的 User 控制 器 中 的 register0 方 法 。register0 方 法 的 关键 代 


码 如 下 : 
a 
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倒 程 09 代码 位 置 资源 包 \TM\10\Shop\Application\Home\Controller\UserController.class.php 


public function register(){ 
// 如 果 已 经 登录 ， 直 接 跳 转 到 首页 
if($this->user_id > 0) header("Location: ".U('Home/Index/index')); 





/表单 提交 操作 **/ 
if(IS_POSTX 
Slogic = new UsersLogic(); /实例 化 逻辑 层 UsersLogic 类 
S$email = I(‘post.email',"); // 接 收 传递 的 邮箱 
$password = I(post.password',"); /接收 传递 的 密码 
$mobile =I(post.mobile',"); /接收 传递 的 手机 号 
$data = $logic->register($email,$password,$mobile); 
/省 略 部 分 代码 
$this->ajaxReturn($data); // 返 回 数据 


} 
/* 非 表单 提交 ， 直 接 显示 页 面 */ 
S$this->display(); 

} 


上 述 代 码 中 ， 首 先 实例 化 逻辑 层 UsersLogic 类 ， 该 类 文件 路 径 为 : Shop\Application\Home\Logic\ 
UsersLogic.class.php。 然 后 ， 调 用 UsersLogic 类 的 register0 方 法 ， 实 现 对 表单 提交 内 容 和 用 户 是 否 已 注 
册 等 信息 进行 验证 。 如 果 全 部 验证 通过 ， 则 将 用 户 信息 写 入 users 表 ， 页 面 跳 转 到 商城 首页 否则 提示 
相应 的 错误 信息 。 


10.6.4 用 户 登 录 功 能 的 实现 过 程 


国 用户 登录 功能 使 用 的 数据 表 : Users 

会 员 登 录 模 块 主要 用 于 实现 网 站 的 会 员 登 录 功 能 ， 在 该 页 面 中 ， 填 写 会 员 账户 和 密码 ， 单 击 “ 登 
录 ” 按 钮 ， 即 可 实现 会 员 登 录 。 如 果 没 有 输入 账户 、 密 码 或 者 账号 密码 不 匹配 ， 都 将 给 予 提示 。 登 录 
模块 功能 主要 是 由 User 控制 器 下 的 do_login0 方 法 实现 的 ， 关 键 代 码 如 下 : 


倒 程 10 代码 位 置 ， 资源 包 \TM\10\Shop\Application\Home\Controller\UserController.class.php 
public function do_login(){ 





$username = trim(I(‘post.username')); // 获 取 用 户 名 并 去 除 首尾 空格 
$password = trim(I(post.password')); // 获 取 密 码 并 去 除 首尾 空格 

Slogic = new UsersLogic(); /实例 化 UsersLogic 类 

Sres = $logic->login($username,$password); // 调 用 UsersLogic 类 的 login() 方 法 
// 省 略 其 余 代码 


} 


上 述 代 码 与 注册 模块 代码 相似 ， 都 是 先 实 例 化 逻辑 层 UsersLogic 类 ， 然 后 调用 登录 验证 的 方法 
login0。 在 login0 方 法 中 , 先 判 断 手机 号 或 者 邮箱 是 否 存 在 , 如 果 存 在 , 继续 判断 输入 的 密码 是 否 正确 ; 
否则 ， 返 回 错误 信息 。 运 行 效果 如 图 10.17 所 示 。 


@ 
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图 10.17 登录 效果 


10.7 ”购物 车 模块 设计 





10.7.1 购物 车 模块 概述 


购物 车 在 51 购 商 城中 是 前 台 客户 端 程序 中 非常 关键 的 一 个 功能 模块 。 购 物 车 的 主要 功能 是 保留 用 
户 选择 的 商品 信息 ， 用 户 可 以 在 购物 车 内 设置 选 购 商品 的 数量 ， 显 示 选 购 商品 的 总 金额 ， 还 可 以 清除 
选择 的 全 部 商品 信息 ， 重 新 选择 商品 信息 。 购 物 车 模块 主要 实现 添加 商品 、 删 除 商品 和 更 改 数量 等 
操作 。 


10.7.2 ”购物 车 模块 技术 分 析 


购物 车 功能 实现 最 关键 的 部 分 就 是 如 何 将 商品 添加 到 购物 车 ， 如 果 不 能 完成 商品 的 添加 ， 那 么 购 
物 车 中 的 其 他 操作 都 没有 任何 意义 。 

在 51 购 商 城 前 台 ， 有 2 个 页 面 可 以 添加 购物 车 。 一 个 是 商品 详情 页 ， 一 个 是 商品 列表 页 。 如 果 在 
商品 详情 页 ， 需 要 先 选择 商品 属性 〈 如 颜色 、 尺 寸 等 )， 然 后 单 击 “ 加 入 购物 车 ”按钮 ， 直 接 加 入 购物 
车 。 如 果 是 在 商品 列表 页 ， 单 击 “ 加 入 购物 车 ”按钮 ， 需 要 判断 该 商品 属性 是 否 存在 ， 如 果 没 有 商品 
属性 (如 图 书 商品 等 )， 可 以 直接 加 入 购物 车 ， 否 则 跳 转 到 商品 详情 页 ， 然 后 再 选择 商品 属性 。 


10.7.3 添加 商品 至 购物 车 的 实现 过 程 


国 添加 商品 至 购物 车 使 用 的 数据 表 : goods、cart 
在 前 台 首 页 选择 商品 后 ， 进 入 商品 详情 页 。 商 品 详情 页 包含 商品 的 名 称 、 简 介 、 属 性 、 价 格 等 信 
息 ， 如 图 10.18 所 示 。 当 选择 完 商品 属性 后 ， 单 击 “ 加 入 购物 车 ”按钮 ， 即 可 将 商品 加 入 购物 车 中 。 运 


行 效果 如 图 10.19 所 示 。 
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图 10.19 加 入 购物 车 


将 商品 加 入 购物 车 的 步 又 如 下 : 
(1) 单 击 “ 加 入 购物 车 ”按钮 ， 执 行 JavaScript 的 onclick0) 单 击 事件 ， 调 用 AjaxAddCart0 方 法 。 


关键 代码 如 下 : 


倒 程 11 代码 位 置 ， 资源 包 \TM\10\Shop\Application\Home\View\Goods\goodsInfo.html 
<!-- 立 即 购买 和 购物 车 start--> 
<div class="join-a-shopping-cart fl" id="join_cart_now"> 
<a class="jrgwc-shopping-img jrgwc-shopping-img2" 
onClick="javascript:AjaxAddCart({$goods.goods_id},1,1):"> 
<span> 立 即 购买 </span> 
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</a> 
</div> 
<div class="join-a-shopping-cart ma-le-210 fl" id="join_cart"> 
<a class="jrgwc-shopping-img2" onClick="javascript:AjaxAddCart({$goods.goods_id},1,0);"> 
<span> 加 入 购物 车 </span> 
</a> 
</div> 
<!-- 立 即 购 买 和 购物 车 end--> 





(2) AjaxAddCart0 是 commonjs 文件 中 的 方法 ， 关 键 代码 如 下 : 
倒 程 12 代码 位 置 ， 资源 包 \TM\10\Shop\Publicjs\common.js 


$.ajax({ 
type : "POST", 
url:"index.php?m=Home&c=Cart&a=ajaxAddCart", 1/ 请求 地 址 
data : $('#buy_goods_form').serialize(), // 搜 索 表单 ， 序 列 化 提交 
dataType:'ijson', /| 数据 格式 
success: function(dataj{ // 请 求 成 功 后 的 响应 
if(data.status < OX{ 
layer.alert(data.msg, {icon: 2}); 
return false; 
} 
// 加 入 购物 车 后 再 跳 转 到 购物 车 页 面 
if(to_catr == 1X{ // 直 接 购买 
location.href = "/index.php?m=Home&c=Cart&a=cart"; // 跳 转 到 购物 车 页 面 
Jelse{ 
cart_num = parselnt($(#cart_quantity ).html())+ 
parselnt($(input[name="goods_num'"]').val()); // 获 取 商 品 数量 
$(#cart_quantity).html(cart_num); /获取 商 品 数量 写 入 DOM 中 
/使 用 layerjs 弹出 加 入 成 功 框 
layer.open({ 
type: 2, // 弹 层 类 型 
tile: 温馨 提示 '， /标题 
skin: "layui-layer-rim', // 弹 层 样式 
area: [490px', '386px]， // 弹 层 宽 高 
content:["/index.php?m=Home&c=Goods&a=open_add_cart","no"],// 页 面 内 容 
success: function(layero, index) { /成 功 响 应 
layer.iframeAuto(index); /指定 iframe 层 自 适应 
} 
»); 
} 
8 


D); 


上 述 代 码 中 , 使 用 Ajax 将 购物 车 中 商品 数据 异步 提交 到 Home 模块 的 Cart 控制 器 的 ajaxAddCartO 
方法 ， 该 方法 主要 用 于 将 商品 信息 写 入 cart 购物 车 表 。 成 功 加 入 购物 车 后 ， 使 用 Layer 弹 层 弹出 加 入 
成 功 页 面 ， 并 且 在 该 页 面 展示 热卖 商品 。 其 中 ， 热 卖 商 品 数据 来 源 于 Goods 控制 器 的 open_add_cartO 


方法 。 
@ 
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10.7.4 查看 购物 车 商品 的 实现 过 程 


国 ” 查 看 购物 车 商品 使 用 的 数据 表 : goods、cart 

将 商品 添加 至 购物 车 后 ， 单 击 “ 去 结算 ”按钮 ， 即 可 跳 转 至 购物 车 列表 页 。 或 者 ,在 前 台 首 页 “我 
的 购物 车 ”按钮 处 显示 购物 车 商品 数量 ， 当 鼠标 悬 停 在 “我 的 购物 车 ”按钮 时 ， 将 展示 所 有 添加 到 购 
物 车 的 商品 ， 如 图 10.20 所 示 。 
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10.20 前 台 首 页 购物 车 效果 


在 购物 车 列表 页 ， 使 用 JavaScript 调用 ajax_cart_list0 方 法 ， 该 方法 使 用 Ajax 异步 提交 的 方式 获取 
购物 车 列表 数据 ， 并 将 获取 到 的 内 容 追 加 到 当前 页 面 。 具 体 代码 如 下 : 


倒 程 13 ”代码 位 置 ， 资源 包 \TM\10\Shop\Home\View\Cart\cart.html 


$(document).ready(function(){ 
ajax_cart_list(); /lajax 请 求 获取 购物 车 列表 
»; 


/ajax 提交 购物 车 
var before_request = 1; /判断 上 一 次 请 求 是 否 已 经 返回 ， 只 有 返回 才 可 以 进行 下 一 次 请 求 
function ajax_cart_list(){ 
if(before_request == 0) // 如 果 上 一 次 请 求 没 返回 ， 则 不 进行 下 一 次 请 求 
return false; 
before_request = 0; 
$.ajax({ 
type : "POST", 
url:"{:U('Home/Cart/ajaxCartList")}", // 提 交 地 址 
data : $('#cart_form').serialize()，”//Form 表单 序列 化 
Success: function(data){ 
S$("#ajax_return").html("); 
$("#ajax_return").append(data); 
before_request = 1; 





: 
六 





} 
@ 
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在 上 述 代码 中 , 将 Ajax 异步 提交 到 Cart 控制 器 下 的 ajaxCartList0 方 法 ， 该 方法 主要 用 于 修改 购物 
车 数量 和 更 改 商 品 选中 状态 ， 并 且 调用 逻辑 层 的 cartLogic 类 的 cartList( 方 法 ， 从 购物 车 表 中 筛选 数据 
并 计算 总 价 。 运 行 效果 如 图 10.21 所 示 。 
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图 10.21 查看 购物 车 效果 


10.7.5 ”清空 购物 车 的 实现 过 程 


国 ”清空 购物 车 使 用 的 数据 表 : goods、cart 

当 用 户 想 要 重新 选 购 商品 时 ， 可 以 删除 购物 车 中 的 某 一 个 商品 或 是 清空 所 有 商品 。 当 单 击 购物 车 
中 某 个 商品 后 的 删除 图 标 ， 弹 出 “确定 要 删除 吗 ? ”确认 框 ， 当 用 户 单 击 “ 确 定 ” 按 钮 时 ， 即 可 删除 
该 商品 。 使 用 Ajax 异步 提交 实现 该 功能 ， 具 体 代 码 如 下 : 


倒 程 14 代码 位 置 资源 包 \TM\10\Shop\Home\View\Cart\cart.html 





/ajax 删除 购物 车 的 商品 
function ajax_del_cart(idsX{ 
layerconfirm(' 确 定 要 删除 吗 ? ,function(){ liLayer 弹 层 确认 框 
$.ajax({ 

type : "POST", /请求 方式 
url:"{:UCHome/Cart/ajaxDelCart)} /请求 地 址 
data:{ids:ids}, /提交 数据 
dataType:json , 


success: function(dataj{ 
if(data.status == 1){ 
layer.msg(data.msg, {icon: 1, time: 1000}, function() { 
ajax_cart_list(); /lajax 请 求 获取 购物 车 列表 
layer.closeAll(); /关闭 弹 层 


六 


难 PHP 项 目 开发 全 程 实录 (第 4 版 ) 





»); 
} 


// 批 量 删除 购物 车 的 商品 
function del_cart_more(){ 
/循环 获 取 复 选 框 选中 的 值 
var chk_value = []; 
S$(input[Iname^="cart_select"]:checked').each(function(X{ 
var s_name = $(this).attr(‘'name'); 
varid = s_name.replace('cart_select[',").replace(J',"); 
chk_value.push(id); 
入 
/ajax 调用 删除 
if(chk_value.length > 0) 
ajax_del_cart(chk_value.join(",")); 
} 


在 上 述 代 码 中 ，ajax_del_cart0 方 法 为 单 选 删除 的 方法 ，del_cart_more0 方 法 为 多 选 删除 的 方法 ， 多 
选 删除 即 是 遍历 单 选 删除 方法 。 接 下 来 ， 看 一 下 Ajax 异步 提交 到 Cart 控制 器 的 ajaxDelCart0 方 法 ， 在 
该 方法 中 实现 从 Cart 数据 表 中 删除 数据 的 操作 。 具 体 代码 如 下 : 


倒 程 15 代码 位 置 ， 资源 包 \TM\10\Shop\Home\Controller\CartController.class.php 
public function ajaxDelCart() 


Sids = I("ids"); /商品 ids 

Sresult = M("Cart")->where(" id in ($ids)")->delete(); 1/ 删除 ids 数组 中 的 数据 
$retum_arr = array('status=>1,msg'=>' 删 除 成 功 ,result=>"); /返回 结果 状态 
Sthis->ajaxReturn($return_arr); 


} 
运行 效果 如 图 10.22 和 图 10.23 所 示 。 





图 10.22 弹 层 删除 提示 框 效果 
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图 10.23 ”删除 成 功效 果 


10.7.6 添加 收 货 地 址 的 实现 过 程 


国 添加 收 货 地 址 使 用 的 数据 表 : user_address、region 
用 户 在 购物 车 列表 页 面 选择 商品 后 ， 单 击 “ 去 结算 ”按钮 ， 跳 转 至 核对 订单 页 面 ， 该 页 面包 括 收 


货 人 信息 和 订单 详细 信息 。 如 果 用 户 首次 购买 商品 ， 单 击 “提交 订单 ”按钮 ， 程 序 会 提示 “请 先 填写 
收 货 人 信息 ”。 运 行 效果 如 图 10.24 所 示 。 


信息 


(NW assesAse 





图 10.24 提示 “请 先 填写 收 货 人 信息 ” 
单 击 图 10.24 中 的 “使 用 新 地 址 ” 超 链接 ， 弹 出 弹 层 ， 显 示 添 加 地 址 的 表单 。 该 功能 主要 是 通过 
Layerjs 插件 来 实现 的 。 关 键 代码 如 下 : 
倒 程 16 ”代码 位 置 ， 资源 包 \TM\10\Shop\Home\View\Cart\step2.html 


<div class="con-y-info ma-bo-35"> 
<h3 style="margin-top:30px"> 收 货 人 信息 <b>[<a href="javascript:void(0);" 
onClick="add_edit_address(0):"> 使 用 新 地 址 </a>]</b></h3> 
<div id="ajax_address"><!--ajax 返回 收 货 地 址 --></div> 


</div> 
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pn 

* 新 增 或 修改 收 货 地 址 

*id 为 0， 为 新 增 , 否则 是 修改 

可 

function add_edit_address(id) 
ifid > OX 


var url = "/index.php?m=Home&c=User&a=edit_address&scene=1 
&call_back=call_back_fun&id="+id; /修改 地 址 


jelse{ 
var url = Windex.php?m=Home&c=User&a=add_address&scene=1 
&call_back=call_back_fun"; /1/ 新 增 地 址 
} 
layer.open({ 
type: 2, // 弹 出 层 类 型 
title: 添加 收 货 地 址 '， /标题 
shadeClose: true, // 是 否 有 遮 淖 层 
shade: 0.3, /阴影 比例 
area: [880px', '580px"], // 弹 层 宽 和 高 
content: url, // 弹 层 内 容 
»; 
} 
运行 效果 如 图 10.25 所 示 。 
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图 10.25” 收 货 地 址 表单 弹 层 


填写 完 收 货 信息 后 ， 单 击 “保存 收 货 地 址 ”按钮 ， 将 调用 User 控制 器 的 add_address0 方 法 ， 该 方 
法 通过 实例 化 逻辑 层 的 UsersLogic 类 ， 调 用 UsersLogic 类 的 add_address() 方 法 ， 实 现 新 增 或 编辑 收 货 
信息 的 功能 。 保 存 成 功 后 ， 运 行 效果 如 图 10.26 所 示 。 


@ 
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图 10.26 ”保存 收 货 信 息 


10.7.7 ”提交 订单 的 实现 过 程 


回 提交 订单 使 用 的 数据 表 : order 

在 核对 订单 页 面 ， 添 加 完 用 户 收 货 信息 后 ， 单 击 “ 提 交 订单 ”按钮 ， 程 序 会 将 订单 信息 写 入 order 
订单 表 。 此 时 ，order 表 中 的 pay_status 〈 订 单 支付 状态 ) 为 0， 表 示 该 订单 未 支付 。 在 “我 的 商城 ”/ 
“我 的 订单 ”可 以 查看 该 订单 ， 如 图 10.27 所 示 。 


我 的 订单 


人 部。 再] 加。 竺 改作 元 


2017.06.19172115 订单 号 ; 2017 
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图 10.27 订单 状态 


单 击 “ 立 即 支付 ”按钮 ， 跳 转 至 订单 支付 页 面 。 该 页 面 中 ， 列 举 了 第 三 方 支付 方式 和 网 银 支 付 方 
式 。 由 于 本 地 测试 无 法 实现 支付 功能 ， 此 外 ， 每 种 支付 方式 的 接口 并 不 相同 ， 读 者 需要 自行 编写 支付 
代码 。 这 里 ， 为 保证 项 目 流程 的 完整 性 ， 使 用 Layer 弹 层 模拟 支付 过 程 。 当 单 击 “ 确 认 支 付 方式 ” 按 


® 
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钮 时 ， 弹 出 支付 弹 层 。 单 击 “ 支 付 ” 按 钮 ， 表 示 支 付 成 功 ， 单 击 “ 取 消 ”按钮 ， 表 示 支 付 失 败 ， 如 
图 10.28 所 示 。 


信息 


想 XI 支 八 过 全 ,点 十 支 何 " 技 要 ， 雪 示 支付 威 功 ， 
高 十 了 内 ,天 元 支 何 舌 玫 





图 10.28 支付 页 面 


支付 成 功 后 ， 在 “我 的 商城 ”/“ 我 的 订单 ”可 以 查看 该 订单 ， 此 时 ， 订 单 状 态 已 经 由 “未 支付 ” 
更 改 为 “ 待 发 货 ”， 如 图 10.29 所 示 。 
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10.29 订单 状态 变更 


10.8 后台 首页 设计 





视频 讲解 
10.8.1 后 台 首 页 概述 


后 台 管理 系统 是 网 站 管理 员 对 商品 、 会 员 及 公告 等 信息 进行 统一 管理 的 场所 ， 本 系统 的 后 台 主 要 


@ 
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包括 以 下 功能 。 

回 ”会 员 管理 模块 : 主要 包括 会 员 管理 和 会 员 等 级 管理 。 其 中 ， 会 员 管理 可 以 实现 对 会 员 的 添加 
和 删除 ， 而 会 员 等 级 管理 可 以 根据 消费 金额 设置 会 员 等 级 。 

回 ”商品 管理 模块 : 主要 包括 商品 分 类 、 商 品 列表 、 商 品类 型 、 商 品 规格 、 商 品 属性 和 商品 评论 。 

回 ”订单 管理 模块 : 主要 包括 查看 订单 、 查 找 订单 和 确认 订单 。 

加 “广告 管理 模块 : 主要 包括 广告 列表 和 广告 位 置 。 

回 “文章 管理 模块 : 主要 包括 文章 分 类 和 文章 列表 。 

后 台 首 页 的 运行 结果 如 图 10.30 所 示 。 


wm 
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本 年 康生 月 全 入 疗 性 杖 加 
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图 1030 后 台 首页 的 运行 结果 
10.8.2 后台 首 页 技术 分 析 


管理 员 登 录 成 功 后 ， 进 入 到 后 台 首页 。 后 台 首 页 是 对 网 站 重要 数据 的 一 个 综合 概述 ， 它 包括 “全 
年 营业 额 ”、“ 全 年 订单 数量 ”、“ 全 部 商品 数量 ”、“ 本 月 会 员 增长 数量 ”和 “月 销售 额 ”等 信息 。 
为 直观 显示 ， 我 们 使 用 卡片 样式 和 柱状 图 形式 展现 以 上 数据 。 其 中 销售 额 柱状 图 ， 是 通过 使 用 百度 
Echarts 插件 实现 的 ， 具 体 使 用 方式 请 参照 官方 文档 : http://echarts.baidu.com。 

后 台 首 页 控制 器 的 关键 代码 如 下 : 


倒 程 17 代码 位 置 ， 资源 包 \TM\10\Shop\Application\Admin\Controller\IndexController.class.php 


public function index() 
‘ 





ww a 1 








/** 获 取 商 品 总 数 */ 

S$total_goods = M('goods')->where(array('is_on_sale'’))->count(); 

$year = date("Y"); 1/ 当 前 年 份 
$month = date("m"); 1/ 当前 月 份 


/* 获 取 最 后 一 个 月 新 增 会 员 **/ 
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S$current_month_start = strtotime($year."-".$month."-01 00:00:00"); /本 月 第 一 天 
$map[reg_time] = array(EGT'.,$current_month_start); 
S$recent_user_number = M('users')->where($map)->count(); 


$start = strtotime($year."-01-01 00:00:00"); /本 年 度 第 一 天 
$end = strtotime($year."-12-31 23:59:59"); /1/ 本 年 度 最 后 一 天 
/SQL 查询 语句 


$sql = "SELECT COUNT(*) as tnum,sum(order_amount) as amount, FROM_UNIXTIME(add_time,'%m') as 
month from 'order "; 
$sql .= " where add_time> $start and add_time <$end and pay_status=1 and order_status in(1,2,4) group 
by month "; 
Sres = M()->query($sql); 
1/ 本 月 订单 总 数 
foreach($res as $arrj{ 
if($arrf'month'] == $month){ /| 当前 月 份 
$current tnum ”= $arr[tnum]; /本 月 销售 数量 
$current_amount = $arrframount]; // 本 月 销售 额 
Jelse{ 
$current_ tnum = 0; 
S$current_amount = 0; 


} 


} 

/* 全 年 销售 数量 和 销售 额 */ 

foreach($res as $arr/{ 
$data[$arr[month]] = $arr[amount]; 
$tnum[$arr[month]] = $arr[tnum']; 


S$all_year tnum ”= array_sum($tnum); // 全 年 销售 数量 
S$all_year_amount = array_sum($data); // 全 年 销售 总 额 


/组织 1-12 月 销售 额 数 据 格式 
for($i=1,$i<13;$i++){ 


if($i < 10X{ 
$i='0.$i 
} 
if(!in_array($i,array_keys($data))}X{ 
S$data[$i] = 0; 
} 
} 
ksort($data); /月份 排序 


foreach($data as $key => $arr}{ 
$result[value]0 = $arr:; 
$resultrmonth']0 = $key; 

} 


S$this->assign('recent_user_number', $recent_user_number); 
Sthis->assign('total_goods", $total_goods); 
S$this->assign('current_tnum', $current_tnum); 
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S$this->assign('current_amount', $current_amount); 
S$this->assign('all_year_tnum',$all_year_tnum); 
S$this->assign('all_year_amount', $all_year_amount); 
Sthis->assign('result'json_encode($result)); 
Sthis->display(); 

} 





接 下 来 演 染 后 台 首 页 模板 ， 关 键 代码 如 下 : 


倒 程 18 ”代码 位 置 ， 资源 包 \TM\10\Shop\ApplicationAdmin\View\Index\index.html 
<extend name="Public/common"/> <!-- 继 承 模板 --> 
<block name="main"> 
<div id="page-wrapper" class="gray-bg"> 
<include file="Public/nav-header"/> 
<div class="wrapper wrapper-content"> 
<div class="row"> 
<div class="col-sm-3"> 
<div class="ibox float-e-margins"> 
<div class="ibox-title"> 
<span class="label label-success pull-right"> 全 年 </span> 
<h5> 营 业 额 </h5> 
</div> 
<div class="ibox-content"> 
<h1 class="no-margins">{$all_year_amount}</h1> 
</div> 
</div> 
</div> 
<div class="col-sm-3"> 
<div class="ibox float-e-margins"> 
<div class="ibox-title"> 
<span class="label label-info pull-right"> 全 年 </span> 
<h5> 订 单 </h5> 
</div> 
<div class="ibox-content"> 
<h1 class="no-margins">{$all_year_tnum}</h1> 
</div> 
</div> 
</div> 
<div class="col-sm-3"> 
<div class="ibox float-e-margins"> 
<div class="ibox-title"> 
<span class="label label-primary pull-right"> 全 部 </span> 
<h5> 商 品 </h5> 
</div> 
<div class="ibox-content"> 
<h1 class="no-margins">{S$total_goods}</h1> 
</div> 
</div> 
</div> 
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<div class="col-sm-3"> 
<div class="ibox float-e-margins"> 
<div class="ibox-title"> 
<span class="label label-danger pull-right"> 最 近 一 个 月 </span> 


<h5> 会 员 </h5> 
</div> 
<div class="ibox-content"> 
<h1 class="no-margins">{$recent_user_number}</h1> 
</div> 
</div> 
</div> 


</div> 
<div class="row"> 
<div class="col-sm-12"> 
<div class="ibox float-e-margins"> 
<div class="ibox-title"> 
<h5> 本 年 度 每 月 销售 额 柱状 
</div> 
<div class="ibox-content"> 
<div class="row"> 
<div class="col-sm-9"> 
<div class="flot-chart" id="main" style="width: 800px;height:400px;"> 





</div> 
</div> 
<div class="col-sm-3" style="margin-top: 100px"> 
<ul class="stat-list"> 
<li> 
<h2 class="no-margins ">{$current_tnum}</h2> 
<small> 本 月 订单 </small> 
<div class="stat-percent"> 
{$current_tnum/$all_year_tnum*100|ceil} % 
</div> 
<div class="progress progress-mini"> 
<div style="width: {$current_tnum/$all_year_tnum*100|ceil}%;" 
Class="progress-bar"></div> 
</div> 
</li> 
<li> 
<h2 class="no-margins ">{$current_amount}</h2> 
<small> 本 月 销售 额 </small> 
<div class="stat-percent"> 
{$current_amount /$all_year_amount * 100|ceil}% 
</div> 
<div class="progress progress-mini"> 
<div style="width: 
{$current_amount/$all_year_amount*100|ceil }%;" 
Class="progress-bar"></div> 
</div> 


[Ga 
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</l> 
</ul> 
</div> 
</div> 
</div> 
</div> 
</div> 
</div> 
</div> 

</div> 

<script src=”_PUBLIC_/plugins/echarts/echarts.minjs"></script> 
<!- 为 ECharts 准备 一 个 具备 大 小 〈 宽 高 ) 的 Dom--> 


<script type="text/javascript"> 
var res = {$result}; 
// 基 于 准备 好 的 dom， 初 始 化 echarts 实例 
var myChart = echarts.init(document.getElementByld('main')); 


// 指 定 图 表 的 配置 项 和 数据 
var option = { 
color: [#3398DB', 
tooltip : { 
trigger: 'axis', 
axisPointer : { /坐标 轴 指 示 器 ， 坐 标 轴 触 发 有 效 
type : 'shadow’ // 默 认为 直线 ， 可 选 为 : ine' | 'shadow 
} 
} 
legend: { 
data:[ 销 售 额 ] 
上 
XAxis:{ 
data: res.month 
入 
yAxis: { 
type : value' 
% 
series: [{ 
name: ' 销 售 额 '， 
type: 'bar’, 
data: res.value 
| 
上 


/使 用 刚 指定 的 配置 项 和 数据 显示 图 表 
myChart.setOption(option); 
</script> 
</block> 
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10.8.3 ”管理 员 登 录 模 块 设计 


国 管理 员 登 录 模 块 使 用 的 数据 表 : admin 

设计 51 购 商 城 时 ， 使 用 了 双 入 口 模式 ， 即 访问 “www.shop.conyindex.php” 进 入 前 台 首 页 ， 访 问 
“www.shop.com/admin.php” 进 入 后 台 首 页 (如果 管理 员 未 登录 ， 则 跳 转 到 后 台 登 录 页 ) 。 在 后 台 登 
录 页 面 中 ， 填 写 管理 员 账 户 、 密 码 和 验证 码 (如 果 验 证 码 看 不 清楚 ， 可 以 单 击 验 证 码 图 片 刷新 该 验证 
码 ) ， 单 击 “ 登 录 ” 按 钮 ， 即 可 实现 管理 员 登录 。 如 果 没 有 输入 账户 、 密 码 或 者 验证 码 ， 都 将 给 予 提 
示 。 另 外 ， 验 证 码 输入 错误 也 将 给 予 提示 。 运 行 效果 如 图 10.31 所 示 。 





图 10.31 后 台 登 录 页 面 


在 后 台 登 录 代 码 中 ， 使 用 ThinkPHP 自身 封装 的 Verify 类 实现 验证 码 的 生成 和 检测 ， 调 用 Verify 
类 的 entry0 方 法 可 以 生成 验证 码 ， 调 用 check0 方 法 可 以 检测 验证 码 。 生 成 验证 码 的 代码 如 下 : 


倒 程 19 ”代码 位 置 : 资源 包 \TM\10\Shop\ApplicationAdmin\Controller\AdminController.class.php 


public function verify(X{ 
S$config= array( 


‘fontSize' => 15, // 验 证 码 字 体 大 小 
"ength' =>4, // 验 证 码 位 数 
"useNoise' => false, /| 关闭 验证 码 杂 点 
'imageW' => 120, /图 片 宽度 

imageH' => 34, /图 片 高 度 

"codeSet =>'0123456789', /随机 产生 0-9 中 的 数字 


); 
$Verify = new\Think\Verify($config); 

$Verify->entry(); // 调 用 entry() 方 法 生成 验证 码 
} 


在 后 台 登 录 页 模板 中 ， 需 要 在 <img> 标 签 内 调用 verify0 方 法 ， 并 且 使 用 JavaScript 的 onClick 单 击 
事件 实现 单 击 图 片 生成 新 验证 码 的 功能 。 关 键 代 码 如 下 : 


@ 
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倒 程 20 ”代码 位 置 资源 包 \TM\10\ShopWApplicationAdmin\View\Admin\login.html 


<div> 
<div> 
<h1 class="logo-name">MR</h1> 
</div> 
<h3> 欢 迎 使 用 51 购 商 城 后 台 </h3> 
<form id="form" name="form" method="post" action=” SELF _” autocomplete="off > 
<div class="form-group"> 
<input name="username" type="text” class="form-control" placeholder=" 用 户 名 " > 
</div> 
<div class="form-group"> 
<input name="password" type="password" class="form-control" placeholder=" 密 码 "> 
</div> 
<div class="form-group login"> 
<span> 验 证 码 </span> 
<input name="code" class="code" type="text" id="code" /> 
<a> <img class="reloadverify" src="{:U('Admin/Admin/verify’)}" id="imgcode" onClick="this.src=this. 
Src+'?'+Math.random()"></a> 
</div> 
<button type="submit" class="btn btn-primary block full-width m-b"> 登 录 </button> 
</form> 
</div> 








10.8.4 商品 模块 设计 


国 商品 模块 使 用 的 数据 表 : goods_category、goods 

商品 模块 是 后 台 最 重要 的 模块 之 一 ， 它 包括 “商品 分 类 ”“ 商 品 列表 ”“ 商 品类 型 ” “商品 规格 ” 
“商品 属性 ”“ 商 品评 论 ”。 在 “商品 列表 ”中 ， 可 以 实现 对 商品 的 增 、 删 、 改 、 查 功能 ， 如 图 10.32 
所 示 。 
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图 10.32 商品 列表 
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在 添加 商品 时 ， 需 要 从 “商品 类 型 ”“ 商 品 规格 ”“ 商 品 属 性 ”中 选择 相应 的 选项 。 为 提高 用 户 


体验 度 ， 使 用 Ajax 异步 提交 的 方式 来 获取 相应 选项 。 此 外 ， 在 添加 “商品 相册 ”时 ， 使 用 了 Plupload 
插件 来 实现 多 图 上 传 功能 。 运 行 效果 如 图 10.33 所 示 。 


青 次 迎 使 用 隔 城 后 会 系 演 人 砂 通 出 
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图 10.33 商品 相册 
10.8.5 ”订单 模块 设计 


国 ”订单 模块 使 用 的 数据 表 : order 

当 用 户 提交 订单 后 ， 在 后 台 “ 订 单 管理 ”/“ 订 单列 表 ” 中 即 可 查看 该 订单 ， 此 时 的 订单 状态 可 能 
是 “未 支付 ”、“ 已 支付 ”或 者 是 “已 作废 ”等 ， 如 图 10.34 所 示 。 管 理 员 需 要 单 击 “ 查 看 ”按钮 ， 查 
看 订单 详情 ， 并 单 击 “确认 ”按钮 确认 订单 。 确 认 无 误 后 ， 在 “订单 列表 ”页 单 击 “发 货 ” 按 钮 ， 弹 
出 确认 框 ， 如 图 10.35 所 示 。 当 单 击 “ 确 定 ” 按 钮 时 ， 使 用 Ajax 异步 更 改 订单 状态 为 “已 发 货 ”， 如 
10.36 所 示 。 


得 欢迎 使 用 而 城 后 台 系 统 。 人 退出 




















图 10.34 未 发 货 状态 
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10.8.6 ”管理 员 模 块 设计 


国 管理 员 模块 使 用 的 数据 表 : admin 

单 击 “ 超 级 管理 员 ” 右 侧 的 下 
拉 图 标 ， 弹 出 “修改 密码 ”和 “ 安 
全 退出 ”选项 框 。 单 击 “ 修 改 密码 ” 


选项 框 ， 即 进入 修改 密码 页 面 。 输 Re 
入 “原始 密码 ”、“ 新 密码 ”和 “ 确 we 
克 认 半 码 : 


认 密 码 ” 后 ， 单 击 “ 提 交 ” 按 钮 ， 
即 可 更 改 密码 ， 如 图 10.37 所 示 。 


10.8.7 会 员 模 块 设计 


国 会员 模块 使 用 的 数据 表 : users、user level 
会 员 模块 包括 “会 员 列 表 ” 和 “会 员 等 级 ”。 当 消费 额度 满足 设置 的 等 级 时 ， 即 可 成 为 该 等 级 的 


会 员 。 会 员 等 级 如 图 10.38 所 示 。 





10.37 修改 密码 


PHP 项 目 开发 全 程 实录 (第 4 版 ) 


清关 病 虹 等 级 扫 述 摊 作 
回回 
人 回回 
区 
天 全 全 回回 
oo000.00 5 四 加 
a 回回 





图 10.38 会 员 等 级 
10.8.8 广告 模块 设计 


国 广告 模块 使 用 的 数据 表 : ad 
广告 模块 包括 “广告 列表 ”和 “广告 位 置 ”。 添 加 广告 时 ， 需 要 先 选择 广告 位 置 。 该 模块 可 以 设 
置 在 不 同 的 前 台 位 置 显 示 不 同 的 广告 内 容 。 广 告 列表 如 图 10.39 所 示 。 
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图 10.39 广告 列表 


10.9 ”开发 技巧 与 难点 分 析 


在 本 系统 开发 和 后 期 测试 的 过 程 中 ,开发 人 员 遇 到 了 各 种 各 样 的 疑难 问题 。 这 里 找 出 一 些 常见 的 、 
容易 被 忽略 的 问题 加 以 讲解 ， 希 望 能 够 为 初学 者 和 新 手提 供 一 些 帮助 ， 在 开发 程序 时 少 走 一 些 弯 路 。 


@ 
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10.9.1 解决 Ajax 的 乱码 问题 


问题 描述 : 当 使 用 Ajax 传递 数据 时 ， 要 么 在 数据 处 理 页 中 数据 不 能 被 正确 处 理 ， 要 么 输出 返回 值 
时 显示 的 是 一 堆 无 法 识别 的 乱码 。 

解决 方法 : 这 是 因为 PHP 在 传递 数据 时 使 用 的 编码 默认 为 UTF-8， 这 就 造成 了 非 英 文字 符 不 能 正 
确 传递 的 情况 ， 开 发 过 程 中 保证 PHP 文件、HTML 文件 均 为 UTF-8g 格式 。 


10.9.2 ”使 用 JavaScript 脚本 获取 、 输 出 标签 内 容 


问题 描述 ， 获取 、 更 改 表单 元 素 值 和 特定 标签 内 容 。 

解决 方法 :使 用 JavaScript 脚本 获取 页 面 内 容 的 方式 主要 有 两 种 ， 第 一 种 是 通过 表单 获取 表单 元 素 
的 value 值 。 格 式 为 : 表单 名 称 .元 素 名 .value。 该 方式 只 能 获取 表单 中 的 元 素 值 ， 对 于 其 他 标签 元 素 则 无 
能 为 力 。 而 第 二 种 方式 可 以 通过 id 名 来 获取 页 面 中 任意 标签 的 内 容 。 格 式 为 “document. getElementById 
(id). value;” 或 “document.getElementById (id).innerText: ”。 

使 用 第 二 种 方式 时 要 注意 , 标签 的 id 名 必须 存在 且 唯 一 ， 否 则 就 会 出 现 错误 。 为 标签 内 容 赋值 时 ， 
则 使 用 如 下 格式 : 


id.innerHTML =' 要 显示 的 内 容 '; 








10.9.3 ”禁用 页 面 缓存 


问题 描述 : 使 用 Ajax 技术 可 以 防止 页 面 刷新 ， 但 有 时 也 会 产生 新 的 问题 。 如 果 进 行 相同 的 操作 ， 
那么 xmlhttprequest 对 象 会 执行 缓存 中 的 信息 ， 从 而 造成 操作 失败 。 

解决 办 法 : 使 用 header0 函 数 将 缓存 关闭 。 将 代码 “header("CACHE-CONTROL:NO-CACHE");” 
添加 到 xmlhttprequest 对 象 所 调用 的 处 理 页 的 顶部 即 可 。 


10.9.4 判断 上 传 文件 格式 


问题 描述 : 添加 商品 时 可 以 上 传 商品 的 图 片 ， 但 有 时 可 能 会 误 传 非 图 片 格式 的 文件 ， 从 而 导致 上 
传 图 片 失败 。 

解决 方法 : ThinkPHP 文件 上 传 操作 使 用 Think\Upload 类 ,通过 设置 exts 属性 ,可 以 控制 上 传 文件 
的 类 型 。 例 如 ， 要 求 上 传 文件 的 类 型 可 以 是 jpg、gif、png， 示 例 代码 如 下 : 


public function upload(){ 


S$upload = newThink\Upload(); /实例 化 上 传 类 
Supload->maxSize = 3145728; // 设 置 附 件 上 传 大 小 
Supload->exts 一) array(jpg,，gif,'png'); // 设 置 附件 上 传 类 型 
S$upload->rootPath = "/Uploads/'; // 设 置 附件 上 传 根 目录 
Supload->savePath = ll 设置 附件 上 传 ( 子 ) 目录 
/上 传 文件 

$info = S$upload->upload(); 


PHP 项 目 开发 全 程 实录 (第 4 版 ) 





if(!$info) { /上 传 错误 提示 错误 信息 
$this->error($upload->getError()): 
jelse{ /1 上 传 成 功 


S$this->success(' 上 传 成 功 ! "); 
} 
让 





10.9.5 ”设置 服务 器 的 时 间 


问题 描述 : 如 果 没有 对 PHP 的 时 区 进行 设置 ， 那 么 使 用 日 期 、 时 间 函 数 获 取 的 将 是 英国 伦敦 本 地 
时 间 〔 即 零 时 区 的 时 间 )。 例 如 ， 以 东 八 区 为 例 ， 如 果 当 地 使 用 的 是 北京 时 间 ， 那 么 如 果 没 有 对 PHP 
的 时 区 进行 设置 ， 获 取 的 时 间 将 比 当 地 的 北京 时 间 少 8 个 小 时 。 

解决 方案 : 要 获取 本 地 当前 的 时 间 必 须 更 改 PHP 语言 中 的 时 区 设置 。 更 改 PHP 语言 中 的 时 区 设置 
有 以 下 两 种 方法 : 

(1) 在 php.ini 文件 中 ， 定 位 到 [date] 下 的 “:date.timezone =” 
选项 ， 去 掉 前 面 的 分 号 ， 并 设置 它 的 值 为 当地 所 在 时 区 使 用 的 时 
间 。 修 改 内 容 如 图 10.40 所 示 。 hh naetyaate tinezone 

例如 ， 如 果 当 地 所 在 时 区 为 东 八 区， 那么 就 可 以 设置 “date. 
timezone=” 的 值 为 PRC、Asia/Hong Kong、Asia/Shanghai (上 海 ) 图 10.40 设置 PHP 的 时 
或 者 Asia/Urumgqi (乌鲁木齐 ) 等 。 这 些 都 是 东 八 区 的 时 间 。 

设置 完成 后 ， 保 存 文件 ， 重 新 启动 Apache 服务 器 。 

(2) 在 应 用 程序 中 ， 在 日 期 、 时 间 函 数 之 前 使 用 date_default_ timezone_setO 函 数 就 可 以 完成 对 时 
区 的 设置 。date_default_timezone_set0 函 数 的 语法 如 下 : 








; Defines the default tinezone used hy the 








多 





date_default_timezone_set(timezone); 


参数 timezone 为 PHP 可 识别 的 时 区 名 称 ， 如 果 时 区 名 称 PHP 无 法 识别 ， 则 系统 采用 UTC 时 区 。 
例如 ， 设 置 北京 时 间 可 以 使 用 的 时 区 包括 PRC (中 华人 民 共 和 国 )、Asia/Chongqing (重庆 )、 
Asia/Shanghai (上 海 ) 或 者 Asia/Urumqi (乌鲁木齐 )， 这 几 个 时 区 名 称 是 等 效 的 。 








10.10 ”ThinkPHP 分 页 技术 专题 


通常 在 数据 查询 后 都 会 对 数据 集 进 行 分 页 操作 ，ThinkPHP 也 提供 了 分 页 类 来 对 数据 分 页 提供 支 
持 。ThinkPHP 分 页 通常 有 如 下 两 种 方式 。 


1. 利用 Page 类 和 limit() 方 法 


limit0 方 法 用 于 获取 特定 数目 的 记录 ， 如 limit(10,25) 表 示 从 第 10 行 开 始 的 25 条 数据 ， 结 合 Page 
类 生成 分 页 的 示例 代码 如 下 : 


S$User = M('User'); /实例 化 User 对 象 
$count = $User->where('status=1)->count(); /查询 满足 要 求 的 总 记录 数 


@ 
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SPage = new\Think\Page($count,25); /实例 化 分 页 类 ， 传 入 总 记录 数 和 每 页 显示 的 记录 数 (25) 
$show = $Page->show(): // 分 页 显示 输出 


/| 进行 分 页 数据 查询 ， 注 意 limit() 方 法 的 参数 要 使 用 Page 类 的 属性 
Slist = $User->where('status=1')->order('create_time')->limit($Page->firstRow.','.$Page->listRows)->select(); 


S$this->assign('list',$list); // 赋 值 数据 集 
Sthis->assign(‘page', $show); // 赋 值 分 页 输出 
S$this->display(); // 输 出 模板 





2. 分 页 类 和 page() 方 法 的 实现 


page0 方 法 也 是 模型 的 连贯 操作 方法 之 一 , 是 完全 为 分 页 查询 而 诞生 的 一 个 人 性 化 操作 方法 .page0 
方法 则 是 更 人 性 化 的 进行 分 页 查询 的 方法 ， 例 如 还 是 以 文章 列表 分 页 为 例 来 说 ， 如 果 使 用 limit0 方 法 ， 
我 们 要 查询 第 一 页 和 第 二 页 〈 假 设 我 们 每 页 输出 10 条 数据 ) ， 代 码 如 下 : 

$Article = M('Article'); 

$Article->limit('0,10')->select(); /| 查询 第 一 页 数据 

S$Article->limit(10,10)->select(); /| 查询 第 二 页 数据 


虽然 利用 扩展 类 库 中 的 分 页 类 Page 可 以 自动 计算 出 每 个 分 页 的 limit 参数 ， 但 是 如 果 要 自己 写 就 
比较 费力 了 ， 如 果 用 page0 方 法 来 写 则 简单 多 了 ， 例 如 : 











$Article = M('Article'); 
S$Article->page('1,10')->select(); /查询 第 一 页 数据 
$Article->page('2,10')->select(); /查询 第 二 页 数据 





显而易见 的 是 ， 使 用 page0 方 法 不 需要 计算 每 个 分 页 数据 的 起 始 位 置 ，page0 方 法 内 部 会 自动 计算 。 
page() 方 法 结合 Page 类 生成 分 页 的 示例 代码 如 下 : 





S$User = M('User'); /实例 化 User 对 象 

// 进 行 分 页 数据 查询 ， 注 意 page() 方 法 的 参数 的 前 面部 分 是 当前 的 页 数 使 用 $_GET[p] 获 取 
Slist = $User->where('status=1')->order('create_time')->page($_GET[p'].',25')->select(); 
Sthis->assign('list', $list); // 赋 值 数据 集 


$count = $User->where('status=1')->count();”// 查 询 满足 要 求 的 总 记录 数 

$Page = new\Think\Page($count,25); /实例 化 分 页 类 ， 传 入 总 记录 数 和 每 页 显示 的 记录 数 
$show = $Page->show(); // 分 页 显示 输出 

Sthis->assign(‘page',$show); /赋值 分 页 输出 

Sthis->display(); // 输 出 模板 





10.11 本 章 总 结 


本 章 主要 介绍 如 何 使 用 ThinkPHP 框架 实现 51 购 商城 项 目 。 包 括 网 站 的 系统 功能 设计 、 数 据 库 设 
计 以 及 前 台 和 后 台 的 主要 功能 模块 。 希 望 通过 本 章 的 学 习 ， 读 者 能 够 将 前 面 章节 所 学 知识 融会 贯通 ， 
了 解 项 目 开发 流程 ， 并 掌握 PHP 网 站 开发 技术 ， 为 今后 的 项 目 开 发 积累 经 验 。 


